Blame gtksourceview/gtksourcefileloader.c

Packit a7d494
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*- */
Packit a7d494
/* gtksourcefileloader.c
Packit a7d494
 * This file is part of GtkSourceView
Packit a7d494
 *
Packit a7d494
 * Copyright (C) 2005 - Paolo Maggi
Packit a7d494
 * Copyright (C) 2007 - Paolo Maggi, 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 "gtksourcefileloader.h"
Packit a7d494
#include "gtksourcebuffer.h"
Packit a7d494
#include "gtksourcefile.h"
Packit a7d494
#include "gtksourcebufferoutputstream.h"
Packit a7d494
#include "gtksourceencoding.h"
Packit a7d494
#include "gtksourceencoding-private.h"
Packit a7d494
#include "gtksourceview-enumtypes.h"
Packit a7d494
#include "gtksourceview-i18n.h"
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * SECTION:fileloader
Packit a7d494
 * @Short_description: Load a file into a GtkSourceBuffer
Packit a7d494
 * @Title: GtkSourceFileLoader
Packit a7d494
 * @See_also: #GtkSourceFile, #GtkSourceFileSaver
Packit a7d494
 *
Packit a7d494
 * A #GtkSourceFileLoader object permits to load the contents of a #GFile or a
Packit a7d494
 * #GInputStream into a #GtkSourceBuffer.
Packit a7d494
 *
Packit a7d494
 * A file loader should be used only for one load operation, including errors
Packit a7d494
 * handling. If an error occurs, you can reconfigure the loader and relaunch the
Packit a7d494
 * operation with gtk_source_file_loader_load_async().
Packit a7d494
 *
Packit a7d494
 * Running a #GtkSourceFileLoader is an undoable action for the
Packit a7d494
 * #GtkSourceBuffer. That is, gtk_source_buffer_begin_not_undoable_action() and
Packit a7d494
 * gtk_source_buffer_end_not_undoable_action() are called, which delete the
Packit a7d494
 * undo/redo history.
Packit a7d494
 *
Packit a7d494
 * After a file loading, the buffer is reset to the contents provided by the
Packit a7d494
 * #GFile or #GInputStream, so the buffer is set as “unmodified”, that is,
Packit a7d494
 * gtk_text_buffer_set_modified() is called with %FALSE. If the contents isn't
Packit a7d494
 * saved somewhere (for example if you load from stdin), then you should
Packit a7d494
 * probably call gtk_text_buffer_set_modified() with %TRUE after calling
Packit a7d494
 * gtk_source_file_loader_load_finish().
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
enum
Packit a7d494
{
Packit a7d494
	PROP_0,
Packit a7d494
	PROP_BUFFER,
Packit a7d494
	PROP_FILE,
Packit a7d494
	PROP_LOCATION,
Packit a7d494
	PROP_INPUT_STREAM
Packit a7d494
};
Packit a7d494
Packit a7d494
#define READ_CHUNK_SIZE 8192
Packit a7d494
#define LOADER_QUERY_ATTRIBUTES G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE "," \
Packit a7d494
				G_FILE_ATTRIBUTE_STANDARD_TYPE "," \
Packit a7d494
				G_FILE_ATTRIBUTE_TIME_MODIFIED "," \
Packit a7d494
				G_FILE_ATTRIBUTE_STANDARD_SIZE "," \
Packit a7d494
				G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE
Packit a7d494
Packit a7d494
struct _GtkSourceFileLoaderPrivate
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 FileLoader.
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 FileLoader.
Packit a7d494
	 */
Packit a7d494
	GtkSourceFile *file;
Packit a7d494
Packit a7d494
	GFile *location;
Packit a7d494
Packit a7d494
	/* The value of the :input-stream property. Do not confuse with the
Packit a7d494
	 * input_stream field in TaskData.
Packit a7d494
	 */
Packit a7d494
	GInputStream *input_stream_property;
Packit a7d494
Packit a7d494
	GSList *candidate_encodings;
Packit a7d494
Packit a7d494
	const GtkSourceEncoding *auto_detected_encoding;
Packit a7d494
	GtkSourceNewlineType auto_detected_newline_type;
Packit a7d494
	GtkSourceCompressionType auto_detected_compression_type;
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 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 output stream, and async
Packit a7d494
	 *     methods for the input stream.
Packit a7d494
	 */
Packit a7d494
	GInputStream *input_stream;
Packit a7d494
	GtkSourceBufferOutputStream *output_stream;
Packit a7d494
Packit a7d494
	GFileInfo *info;
Packit a7d494
Packit a7d494
	GFileProgressCallback progress_cb;
Packit a7d494
	gpointer progress_cb_data;
Packit a7d494
	GDestroyNotify progress_cb_notify;
Packit a7d494
Packit a7d494
	goffset total_bytes_read;
Packit a7d494
	goffset total_size;
Packit a7d494
Packit a7d494
	gssize chunk_bytes_read;
Packit a7d494
	gchar chunk_buffer[READ_CHUNK_SIZE];
Packit a7d494
Packit a7d494
	guint guess_content_type_from_content : 1;
Packit a7d494
	guint tried_mount : 1;
Packit a7d494
};
Packit a7d494
Packit a7d494
G_DEFINE_TYPE_WITH_PRIVATE (GtkSourceFileLoader, gtk_source_file_loader, G_TYPE_OBJECT)
Packit a7d494
Packit a7d494
static void open_file (GTask *task);
Packit a7d494
static void read_file_chunk (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
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 GtkSourceCompressionType
Packit a7d494
get_compression_type_from_content_type (const gchar *content_type)
Packit a7d494
{
Packit a7d494
	if (content_type == NULL)
Packit a7d494
	{
Packit a7d494
		return GTK_SOURCE_COMPRESSION_TYPE_NONE;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (g_content_type_is_a (content_type, "application/x-gzip"))
Packit a7d494
	{
Packit a7d494
		return GTK_SOURCE_COMPRESSION_TYPE_GZIP;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return GTK_SOURCE_COMPRESSION_TYPE_NONE;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_loader_set_property (GObject      *object,
Packit a7d494
				     guint         prop_id,
Packit a7d494
				     const GValue *value,
Packit a7d494
				     GParamSpec   *pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader = GTK_SOURCE_FILE_LOADER (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_BUFFER:
Packit a7d494
			g_assert (loader->priv->source_buffer == NULL);
Packit a7d494
			loader->priv->source_buffer = g_value_get_object (value);
Packit a7d494
			g_object_add_weak_pointer (G_OBJECT (loader->priv->source_buffer),
Packit a7d494
						   (gpointer *)&loader->priv->source_buffer);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_FILE:
Packit a7d494
			g_assert (loader->priv->file == NULL);
Packit a7d494
			loader->priv->file = g_value_get_object (value);
Packit a7d494
			g_object_add_weak_pointer (G_OBJECT (loader->priv->file),
Packit a7d494
						   (gpointer *)&loader->priv->file);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_LOCATION:
Packit a7d494
			g_assert (loader->priv->location == NULL);
Packit a7d494
			loader->priv->location = g_value_dup_object (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_INPUT_STREAM:
Packit a7d494
			g_assert (loader->priv->input_stream_property == NULL);
Packit a7d494
			loader->priv->input_stream_property = g_value_dup_object (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_loader_get_property (GObject    *object,
Packit a7d494
				     guint       prop_id,
Packit a7d494
				     GValue     *value,
Packit a7d494
				     GParamSpec *pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader = GTK_SOURCE_FILE_LOADER (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_BUFFER:
Packit a7d494
			g_value_set_object (value, loader->priv->source_buffer);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_FILE:
Packit a7d494
			g_value_set_object (value, loader->priv->file);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_LOCATION:
Packit a7d494
			g_value_set_object (value, loader->priv->location);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_INPUT_STREAM:
Packit a7d494
			g_value_set_object (value, loader->priv->input_stream_property);
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_loader_dispose (GObject *object)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader = GTK_SOURCE_FILE_LOADER (object);
Packit a7d494
Packit a7d494
	if (loader->priv->source_buffer != NULL)
Packit a7d494
	{
Packit a7d494
		g_object_remove_weak_pointer (G_OBJECT (loader->priv->source_buffer),
Packit a7d494
					      (gpointer *)&loader->priv->source_buffer);
Packit a7d494
Packit a7d494
		loader->priv->source_buffer = NULL;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (loader->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		g_object_remove_weak_pointer (G_OBJECT (loader->priv->file),
Packit a7d494
					      (gpointer *)&loader->priv->file);
Packit a7d494
Packit a7d494
		loader->priv->file = NULL;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_clear_object (&loader->priv->location);
Packit a7d494
	g_clear_object (&loader->priv->input_stream_property);
Packit a7d494
	g_clear_object (&loader->priv->task);
Packit a7d494
Packit a7d494
	g_slist_free (loader->priv->candidate_encodings);
Packit a7d494
	loader->priv->candidate_encodings = NULL;
Packit a7d494
Packit a7d494
	G_OBJECT_CLASS (gtk_source_file_loader_parent_class)->dispose (object);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
set_default_candidate_encodings (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	GSList *list;
Packit a7d494
	GSList *l;
Packit a7d494
	const GtkSourceEncoding *file_encoding;
Packit a7d494
Packit a7d494
	/* Get first the default candidates from GtkSourceEncoding. If the
Packit a7d494
	 * GtkSourceFile's encoding has been set by a FileLoader or FileSaver,
Packit a7d494
	 * put it at the beginning of the list.
Packit a7d494
	 */
Packit a7d494
	list = gtk_source_encoding_get_default_candidates ();
Packit a7d494
Packit a7d494
	if (loader->priv->file == NULL)
Packit a7d494
	{
Packit a7d494
		goto end;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	file_encoding = gtk_source_file_get_encoding (loader->priv->file);
Packit a7d494
Packit a7d494
	if (file_encoding == NULL)
Packit a7d494
	{
Packit a7d494
		goto end;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* Remove file_encoding from the list, if already present, and prepend
Packit a7d494
	 * it to the list.
Packit a7d494
	 */
Packit a7d494
	for (l = list; l != NULL; l = l->next)
Packit a7d494
	{
Packit a7d494
		const GtkSourceEncoding *cur_encoding = l->data;
Packit a7d494
Packit a7d494
		if (cur_encoding == file_encoding)
Packit a7d494
		{
Packit a7d494
			list = g_slist_delete_link (list, l);
Packit a7d494
Packit a7d494
			/* The list doesn't contain duplicates, normally. */
Packit a7d494
			break;
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	list = g_slist_prepend (list, (gpointer) file_encoding);
Packit a7d494
Packit a7d494
end:
Packit a7d494
	g_slist_free (loader->priv->candidate_encodings);
Packit a7d494
	loader->priv->candidate_encodings = list;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_loader_constructed (GObject *object)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader = GTK_SOURCE_FILE_LOADER (object);
Packit a7d494
Packit a7d494
	if (loader->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		set_default_candidate_encodings (loader);
Packit a7d494
Packit a7d494
		if (loader->priv->location == NULL &&
Packit a7d494
		    loader->priv->input_stream_property == NULL)
Packit a7d494
		{
Packit a7d494
			loader->priv->location = gtk_source_file_get_location (loader->priv->file);
Packit a7d494
Packit a7d494
			if (loader->priv->location != NULL)
Packit a7d494
			{
Packit a7d494
				g_object_ref (loader->priv->location);
Packit a7d494
			}
Packit a7d494
			else
Packit a7d494
			{
Packit a7d494
				g_warning ("GtkSourceFileLoader: the GtkSourceFile's location is NULL. "
Packit a7d494
					   "Call gtk_source_file_set_location() or read from a GInputStream.");
Packit a7d494
			}
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	G_OBJECT_CLASS (gtk_source_file_loader_parent_class)->constructed (object);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_loader_class_init (GtkSourceFileLoaderClass *klass)
Packit a7d494
{
Packit a7d494
	GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit a7d494
Packit a7d494
	object_class->dispose = gtk_source_file_loader_dispose;
Packit a7d494
	object_class->set_property = gtk_source_file_loader_set_property;
Packit a7d494
	object_class->get_property = gtk_source_file_loader_get_property;
Packit a7d494
	object_class->constructed = gtk_source_file_loader_constructed;
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileLoader:buffer:
Packit a7d494
	 *
Packit a7d494
	 * The #GtkSourceBuffer to load the contents into. The
Packit a7d494
	 * #GtkSourceFileLoader object has a weak reference to the buffer.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class, 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
	 * GtkSourceFileLoader:file:
Packit a7d494
	 *
Packit a7d494
	 * The #GtkSourceFile. The #GtkSourceFileLoader 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, 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
	 * GtkSourceFileLoader:location:
Packit a7d494
	 *
Packit a7d494
	 * The #GFile to load. If the #GtkSourceFileLoader:input-stream is
Packit a7d494
	 * %NULL, by default the location is taken from the #GtkSourceFile at
Packit a7d494
	 * construction time.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class, 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
	 * GtkSourceFileLoader:input-stream:
Packit a7d494
	 *
Packit a7d494
	 * The #GInputStream to load. Useful for reading stdin. If this property
Packit a7d494
	 * is set, the #GtkSourceFileLoader:location property is ignored.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class, PROP_INPUT_STREAM,
Packit a7d494
					 g_param_spec_object ("input-stream",
Packit a7d494
							      "Input stream",
Packit a7d494
							      "",
Packit a7d494
							      G_TYPE_INPUT_STREAM,
Packit a7d494
							      G_PARAM_READWRITE |
Packit a7d494
							      G_PARAM_CONSTRUCT_ONLY |
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 GtkSourceBufferOutputStream
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_OUTPUT_STREAM);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_loader_init (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	loader->priv = gtk_source_file_loader_get_instance_private (loader);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
close_input_stream_cb (GObject      *source_object,
Packit a7d494
		       GAsyncResult *result,
Packit a7d494
		       gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GInputStream *input_stream = G_INPUT_STREAM (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 ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_input_stream_close_finish (input_stream, result, &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_task_return_error (task, error);
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 (G_OUTPUT_STREAM (task_data->output_stream),
Packit a7d494
			       g_task_get_cancellable (task),
Packit a7d494
			       &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 if we needed some fallback char, if so, check if there was a
Packit a7d494
	 * previous error and if not set a fallback used error.
Packit a7d494
	 */
Packit a7d494
	if (gtk_source_buffer_output_stream_get_num_fallbacks (task_data->output_stream) > 0)
Packit a7d494
	{
Packit a7d494
		g_task_return_new_error (task,
Packit a7d494
					 GTK_SOURCE_FILE_LOADER_ERROR,
Packit a7d494
					 GTK_SOURCE_FILE_LOADER_ERROR_CONVERSION_FALLBACK,
Packit a7d494
					 _("There was a character encoding conversion error "
Packit a7d494
					   "and it was needed to use a fallback character."));
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
write_complete (GTask *task)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_input_stream_close_async (task_data->input_stream,
Packit a7d494
				    g_task_get_priority (task),
Packit a7d494
				    g_task_get_cancellable (task),
Packit a7d494
				    close_input_stream_cb,
Packit a7d494
				    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
	gssize chunk_bytes_written = 0;
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	while (chunk_bytes_written < task_data->chunk_bytes_read)
Packit a7d494
	{
Packit a7d494
		gssize bytes_written;
Packit a7d494
		GError *error = NULL;
Packit a7d494
Packit a7d494
		/* We use sync methods on the buffer stream since it is in memory. Using
Packit a7d494
		 * async would be racy and we can end up with invalidated iters.
Packit a7d494
		 */
Packit a7d494
		bytes_written = g_output_stream_write (G_OUTPUT_STREAM (task_data->output_stream),
Packit a7d494
						       task_data->chunk_buffer + chunk_bytes_written,
Packit a7d494
						       task_data->chunk_bytes_read - chunk_bytes_written,
Packit a7d494
						       g_task_get_cancellable (task),
Packit a7d494
						       &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_task_return_error (task, error);
Packit a7d494
			return;
Packit a7d494
		}
Packit a7d494
Packit a7d494
		chunk_bytes_written += bytes_written;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* FIXME: note that calling the progress callback blocks the read...
Packit a7d494
	 * Check if it isn't a performance problem.
Packit a7d494
	 */
Packit a7d494
	if (task_data->progress_cb != NULL &&
Packit a7d494
	    task_data->total_size > 0)
Packit a7d494
	{
Packit a7d494
		task_data->progress_cb (task_data->total_bytes_read,
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
read_cb (GObject      *source_object,
Packit a7d494
	 GAsyncResult *result,
Packit a7d494
	 gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GInputStream *input_stream = G_INPUT_STREAM (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	GtkSourceFileLoader *loader;
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
	loader = g_task_get_source_object (task);
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	task_data->chunk_bytes_read = g_input_stream_read_finish (input_stream, 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 for the extremely unlikely case where the file size overflows. */
Packit a7d494
	if (task_data->total_bytes_read + task_data->chunk_bytes_read < task_data->total_bytes_read)
Packit a7d494
	{
Packit a7d494
		g_task_return_new_error (task,
Packit a7d494
					 GTK_SOURCE_FILE_LOADER_ERROR,
Packit a7d494
					 GTK_SOURCE_FILE_LOADER_ERROR_TOO_BIG,
Packit a7d494
					 _("File too big."));
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (task_data->guess_content_type_from_content &&
Packit a7d494
	    task_data->chunk_bytes_read > 0 &&
Packit a7d494
	    task_data->total_bytes_read == 0)
Packit a7d494
	{
Packit a7d494
		gchar *guessed;
Packit a7d494
Packit a7d494
		guessed = g_content_type_guess (NULL,
Packit a7d494
		                                (guchar *)task_data->chunk_buffer,
Packit a7d494
		                                task_data->chunk_bytes_read,
Packit a7d494
		                                NULL);
Packit a7d494
Packit a7d494
		if (guessed != NULL)
Packit a7d494
		{
Packit a7d494
			g_file_info_set_attribute_string (task_data->info,
Packit a7d494
			                                  G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE,
Packit a7d494
			                                  guessed);
Packit a7d494
Packit a7d494
			g_free (guessed);
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* End of the file, we are done! */
Packit a7d494
	if (task_data->chunk_bytes_read == 0)
Packit a7d494
	{
Packit a7d494
		/* Flush the stream to ensure proper line ending detection. */
Packit a7d494
		g_output_stream_flush (G_OUTPUT_STREAM (task_data->output_stream), NULL, NULL);
Packit a7d494
Packit a7d494
		loader->priv->auto_detected_encoding =
Packit a7d494
			gtk_source_buffer_output_stream_get_guessed (task_data->output_stream);
Packit a7d494
Packit a7d494
		loader->priv->auto_detected_newline_type =
Packit a7d494
			gtk_source_buffer_output_stream_detect_newline_type (task_data->output_stream);
Packit a7d494
Packit a7d494
		write_complete (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	task_data->total_bytes_read += task_data->chunk_bytes_read;
Packit a7d494
Packit a7d494
	write_file_chunk (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
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_input_stream_read_async (task_data->input_stream,
Packit a7d494
				   task_data->chunk_buffer,
Packit a7d494
				   READ_CHUNK_SIZE,
Packit a7d494
				   g_task_get_priority (task),
Packit a7d494
				   g_task_get_cancellable (task),
Packit a7d494
				   read_cb,
Packit a7d494
				   task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
add_gzip_decompressor_stream (GTask *task)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GZlibDecompressor *decompressor;
Packit a7d494
	GInputStream *new_input_stream;
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	decompressor = g_zlib_decompressor_new (G_ZLIB_COMPRESSOR_FORMAT_GZIP);
Packit a7d494
Packit a7d494
	new_input_stream = g_converter_input_stream_new (task_data->input_stream,
Packit a7d494
							 G_CONVERTER (decompressor));
Packit a7d494
Packit a7d494
	g_object_unref (task_data->input_stream);
Packit a7d494
	g_object_unref (decompressor);
Packit a7d494
Packit a7d494
	task_data->input_stream = new_input_stream;
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
create_input_stream (GTask *task)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader;
Packit a7d494
	TaskData *task_data;
Packit a7d494
Packit a7d494
	loader = g_task_get_source_object (task);
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	loader->priv->auto_detected_compression_type = GTK_SOURCE_COMPRESSION_TYPE_NONE;
Packit a7d494
Packit a7d494
	if (loader->priv->input_stream_property != NULL)
Packit a7d494
	{
Packit a7d494
		task_data->input_stream = g_object_ref (loader->priv->input_stream_property);
Packit a7d494
	}
Packit a7d494
	else if (g_file_info_has_attribute (task_data->info, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE))
Packit a7d494
	{
Packit a7d494
		const gchar *content_type = g_file_info_get_content_type (task_data->info);
Packit a7d494
Packit a7d494
		switch (get_compression_type_from_content_type (content_type))
Packit a7d494
		{
Packit a7d494
			case GTK_SOURCE_COMPRESSION_TYPE_GZIP:
Packit a7d494
				add_gzip_decompressor_stream (task);
Packit a7d494
				loader->priv->auto_detected_compression_type = GTK_SOURCE_COMPRESSION_TYPE_GZIP;
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			case GTK_SOURCE_COMPRESSION_TYPE_NONE:
Packit a7d494
				/* NOOP */
Packit a7d494
				break;
Packit a7d494
Packit a7d494
			default:
Packit a7d494
				g_assert_not_reached ();
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_return_if_fail (task_data->input_stream != NULL);
Packit a7d494
Packit a7d494
	/* start reading */
Packit a7d494
	read_file_chunk (task);
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 ("%s\n", G_STRFUNC);
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
		g_task_return_error (task, error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (g_file_info_has_attribute (task_data->info, G_FILE_ATTRIBUTE_STANDARD_TYPE) &&
Packit a7d494
	    g_file_info_get_file_type (task_data->info) != G_FILE_TYPE_REGULAR)
Packit a7d494
	{
Packit a7d494
		g_task_return_new_error (task,
Packit a7d494
					 G_IO_ERROR,
Packit a7d494
					 G_IO_ERROR_NOT_REGULAR_FILE,
Packit a7d494
					 _("Not a regular file."));
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (g_file_info_has_attribute (task_data->info, G_FILE_ATTRIBUTE_STANDARD_SIZE))
Packit a7d494
	{
Packit a7d494
		task_data->total_size = g_file_info_get_attribute_uint64 (task_data->info,
Packit a7d494
									  G_FILE_ATTRIBUTE_STANDARD_SIZE);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	create_input_stream (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
	}
Packit a7d494
	else
Packit a7d494
	{
Packit a7d494
		/* Try again to open the file for reading. */
Packit a7d494
		open_file (task);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
recover_not_mounted (GTask *task)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader;
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GMountOperation *mount_operation;
Packit a7d494
Packit a7d494
	loader = 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 (loader->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 (loader->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
static void
Packit a7d494
open_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
	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
	g_clear_object (&task_data->input_stream);
Packit a7d494
	task_data->input_stream = G_INPUT_STREAM (g_file_read_finish (location, result, &error));
Packit a7d494
Packit a7d494
	if (error != NULL)
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
		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_input_stream_query_info_async since it is not able to get the
Packit a7d494
	 * content type etc, beside it is not supported by gvfs.
Packit a7d494
	 * Using the file instead of the stream is slightly racy, but for
Packit a7d494
	 * loading this is not too bad...
Packit a7d494
	 */
Packit a7d494
	g_file_query_info_async (location,
Packit a7d494
				 LOADER_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
open_file (GTask *task)
Packit a7d494
{
Packit a7d494
	GtkSourceFileLoader *loader;
Packit a7d494
Packit a7d494
	loader = g_task_get_source_object (task);
Packit a7d494
Packit a7d494
	g_file_read_async (loader->priv->location,
Packit a7d494
	                   g_task_get_priority (task),
Packit a7d494
			   g_task_get_cancellable (task),
Packit a7d494
	                   open_file_cb,
Packit a7d494
	                   task);
Packit a7d494
}
Packit a7d494
Packit a7d494
GQuark
Packit a7d494
gtk_source_file_loader_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-loader-error");
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return quark;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_new:
Packit a7d494
 * @buffer: the #GtkSourceBuffer to load the contents into.
Packit a7d494
 * @file: the #GtkSourceFile.
Packit a7d494
 *
Packit a7d494
 * Creates a new #GtkSourceFileLoader object. The contents is read from the
Packit a7d494
 * #GtkSourceFile's location. If not already done, call
Packit a7d494
 * gtk_source_file_set_location() before calling this constructor. The previous
Packit a7d494
 * location is anyway not needed, because as soon as the file loading begins,
Packit a7d494
 * the @buffer is emptied.
Packit a7d494
 *
Packit a7d494
 * Returns: a new #GtkSourceFileLoader object.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFileLoader *
Packit a7d494
gtk_source_file_loader_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_LOADER,
Packit a7d494
			     "buffer", buffer,
Packit a7d494
			     "file", file,
Packit a7d494
			     NULL);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_new_from_stream:
Packit a7d494
 * @buffer: the #GtkSourceBuffer to load the contents into.
Packit a7d494
 * @file: the #GtkSourceFile.
Packit a7d494
 * @stream: the #GInputStream to load, e.g. stdin.
Packit a7d494
 *
Packit a7d494
 * Creates a new #GtkSourceFileLoader object. The contents is read from @stream.
Packit a7d494
 *
Packit a7d494
 * Returns: a new #GtkSourceFileLoader object.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFileLoader *
Packit a7d494
gtk_source_file_loader_new_from_stream (GtkSourceBuffer *buffer,
Packit a7d494
					GtkSourceFile   *file,
Packit a7d494
					GInputStream    *stream)
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_INPUT_STREAM (stream), NULL);
Packit a7d494
Packit a7d494
	return g_object_new (GTK_SOURCE_TYPE_FILE_LOADER,
Packit a7d494
			     "buffer", buffer,
Packit a7d494
			     "file", file,
Packit a7d494
			     "input-stream", stream,
Packit a7d494
			     NULL);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_set_candidate_encodings:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 * @candidate_encodings: (element-type GtkSourceEncoding): a list of
Packit a7d494
 *   #GtkSourceEncodings.
Packit a7d494
 *
Packit a7d494
 * Sets the candidate encodings for the file loading. The encodings are tried in
Packit a7d494
 * the same order as the list.
Packit a7d494
 *
Packit a7d494
 * For convenience, @candidate_encodings can contain duplicates. Only the first
Packit a7d494
 * occurrence of a duplicated encoding is kept in the list.
Packit a7d494
 *
Packit a7d494
 * By default the candidate encodings are (in that order in the list):
Packit a7d494
 * 1. If set, the #GtkSourceFile's encoding as returned by
Packit a7d494
 * gtk_source_file_get_encoding().
Packit a7d494
 * 2. The default candidates as returned by
Packit a7d494
 * gtk_source_encoding_get_default_candidates().
Packit a7d494
 *
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_file_loader_set_candidate_encodings (GtkSourceFileLoader *loader,
Packit a7d494
						GSList              *candidate_encodings)
Packit a7d494
{
Packit a7d494
	GSList *list;
Packit a7d494
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader));
Packit a7d494
	g_return_if_fail (loader->priv->task == NULL);
Packit a7d494
Packit a7d494
	list = g_slist_copy (candidate_encodings);
Packit a7d494
	list = _gtk_source_encoding_remove_duplicates (list, GTK_SOURCE_ENCODING_DUPLICATES_KEEP_FIRST);
Packit a7d494
Packit a7d494
	g_slist_free (loader->priv->candidate_encodings);
Packit a7d494
	loader->priv->candidate_encodings = list;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_buffer:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: (transfer none): the #GtkSourceBuffer to load the contents into.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceBuffer *
Packit a7d494
gtk_source_file_loader_get_buffer (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader), NULL);
Packit a7d494
Packit a7d494
	return loader->priv->source_buffer;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_file:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: (transfer none): the #GtkSourceFile.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFile *
Packit a7d494
gtk_source_file_loader_get_file (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader), NULL);
Packit a7d494
Packit a7d494
	return loader->priv->file;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_location:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: (nullable) (transfer none): the #GFile to load, or %NULL
Packit a7d494
 * if an input stream is used.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GFile *
Packit a7d494
gtk_source_file_loader_get_location (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader), NULL);
Packit a7d494
Packit a7d494
	return loader->priv->location;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_input_stream:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: (nullable) (transfer none): the #GInputStream to load, or %NULL
Packit a7d494
 * if a #GFile is used.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GInputStream *
Packit a7d494
gtk_source_file_loader_get_input_stream (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader), NULL);
Packit a7d494
Packit a7d494
	return loader->priv->input_stream_property;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_load_async:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
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
 * Loads asynchronously the file or input stream contents into the
Packit a7d494
 * #GtkSourceBuffer. See the #GAsyncResult documentation to know how to use this
Packit a7d494
 * 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_loader_load_async (GtkSourceFileLoader   *loader,
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 implicit_trailing_newline;
Packit a7d494
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader));
Packit a7d494
	g_return_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable));
Packit a7d494
	g_return_if_fail (loader->priv->task == NULL);
Packit a7d494
Packit a7d494
	loader->priv->task = g_task_new (loader, cancellable, callback, user_data);
Packit a7d494
	g_task_set_priority (loader->priv->task, io_priority);
Packit a7d494
Packit a7d494
	task_data = task_data_new ();
Packit a7d494
	g_task_set_task_data (loader->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 (loader->priv->source_buffer == NULL ||
Packit a7d494
	    loader->priv->file == NULL ||
Packit a7d494
	    (loader->priv->location == NULL && loader->priv->input_stream_property == NULL))
Packit a7d494
	{
Packit a7d494
		g_task_return_boolean (loader->priv->task, FALSE);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Start loading\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	/* Update GtkSourceFile location directly. The other GtkSourceFile
Packit a7d494
	 * properties are updated when the operation is finished. But since the
Packit a7d494
	 * file is loaded, the previous contents is lost, so the previous
Packit a7d494
	 * location is anyway not needed. And for display purposes, the new
Packit a7d494
	 * location is directly needed (for example to display the filename in a
Packit a7d494
	 * tab or an info bar with the progress information).
Packit a7d494
	 */
Packit a7d494
	if (loader->priv->input_stream_property != NULL)
Packit a7d494
	{
Packit a7d494
		gtk_source_file_set_location (loader->priv->file, NULL);
Packit a7d494
	}
Packit a7d494
	else
Packit a7d494
	{
Packit a7d494
		gtk_source_file_set_location (loader->priv->file,
Packit a7d494
					      loader->priv->location);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	implicit_trailing_newline = gtk_source_buffer_get_implicit_trailing_newline (loader->priv->source_buffer);
Packit a7d494
Packit a7d494
	/* The BufferOutputStream has a strong reference to the buffer.
Packit a7d494
         * We create the BufferOutputStream here so we are sure that the
Packit a7d494
         * buffer will not be destroyed during the file loading.
Packit a7d494
         */
Packit a7d494
	task_data->output_stream = gtk_source_buffer_output_stream_new (loader->priv->source_buffer,
Packit a7d494
									loader->priv->candidate_encodings,
Packit a7d494
									implicit_trailing_newline);
Packit a7d494
Packit a7d494
	if (loader->priv->input_stream_property != NULL)
Packit a7d494
	{
Packit a7d494
		task_data->guess_content_type_from_content = TRUE;
Packit a7d494
		task_data->info = g_file_info_new ();
Packit a7d494
Packit a7d494
		create_input_stream (loader->priv->task);
Packit a7d494
	}
Packit a7d494
	else
Packit a7d494
	{
Packit a7d494
		open_file (loader->priv->task);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_load_finish:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 * @result: a #GAsyncResult.
Packit a7d494
 * @error: a #GError, or %NULL.
Packit a7d494
 *
Packit a7d494
 * Finishes a file loading started with gtk_source_file_loader_load_async().
Packit a7d494
 *
Packit a7d494
 * If the contents has been loaded, the following #GtkSourceFile properties will
Packit a7d494
 * be updated: the location, the encoding, the newline type and the compression
Packit a7d494
 * type.
Packit a7d494
 *
Packit a7d494
 * Returns: whether the contents has been loaded successfully.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
gboolean
Packit a7d494
gtk_source_file_loader_load_finish (GtkSourceFileLoader  *loader,
Packit a7d494
				    GAsyncResult         *result,
Packit a7d494
				    GError              **error)
Packit a7d494
{
Packit a7d494
	gboolean ok;
Packit a7d494
	gboolean update_file_properties;
Packit a7d494
	GError *real_error = NULL;
Packit a7d494
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader), 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, loader), FALSE);
Packit a7d494
Packit a7d494
	ok = g_task_propagate_boolean (G_TASK (result), &real_error);
Packit a7d494
Packit a7d494
	if (error != NULL && real_error != NULL)
Packit a7d494
	{
Packit a7d494
		*error = g_error_copy (real_error);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* Update the file properties if the contents has been loaded. The
Packit a7d494
	 * contents can be loaded successfully, or there can be encoding
Packit a7d494
	 * conversion errors with fallback characters. In the latter case, the
Packit a7d494
	 * encoding may be wrong, but since the contents has anyway be loaded,
Packit a7d494
	 * the file properties must be updated.
Packit a7d494
	 * With the other errors, normally the contents hasn't been loaded into
Packit a7d494
	 * the buffer, i.e. the buffer is still empty.
Packit a7d494
	 */
Packit a7d494
	update_file_properties = ok || (real_error != NULL &&
Packit a7d494
					real_error->domain == GTK_SOURCE_FILE_LOADER_ERROR &&
Packit a7d494
					real_error->code == GTK_SOURCE_FILE_LOADER_ERROR_CONVERSION_FALLBACK);
Packit a7d494
Packit a7d494
	if (update_file_properties && loader->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		TaskData *task_data;
Packit a7d494
Packit a7d494
		task_data = g_task_get_task_data (G_TASK (result));
Packit a7d494
Packit a7d494
		/* The location is already updated at the beginning of the
Packit a7d494
		 * operation.
Packit a7d494
		 */
Packit a7d494
Packit a7d494
		_gtk_source_file_set_encoding (loader->priv->file,
Packit a7d494
					       loader->priv->auto_detected_encoding);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_newline_type (loader->priv->file,
Packit a7d494
						   loader->priv->auto_detected_newline_type);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_compression_type (loader->priv->file,
Packit a7d494
						       loader->priv->auto_detected_compression_type);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_externally_modified (loader->priv->file, FALSE);
Packit a7d494
		_gtk_source_file_set_deleted (loader->priv->file, FALSE);
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 (loader->priv->file, modification_time);
Packit a7d494
		}
Packit a7d494
Packit a7d494
		if (g_file_info_has_attribute (task_data->info, G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE))
Packit a7d494
		{
Packit a7d494
			gboolean readonly;
Packit a7d494
Packit a7d494
			readonly = !g_file_info_get_attribute_boolean (task_data->info,
Packit a7d494
								       G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE);
Packit a7d494
Packit a7d494
			_gtk_source_file_set_readonly (loader->priv->file, readonly);
Packit a7d494
		}
Packit a7d494
		else
Packit a7d494
		{
Packit a7d494
			_gtk_source_file_set_readonly (loader->priv->file, FALSE);
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_clear_object (&loader->priv->task);
Packit a7d494
Packit a7d494
	if (real_error != NULL)
Packit a7d494
	{
Packit a7d494
		g_error_free (real_error);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return ok;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_encoding:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: the detected file encoding.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
const GtkSourceEncoding *
Packit a7d494
gtk_source_file_loader_get_encoding (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader), NULL);
Packit a7d494
Packit a7d494
	return loader->priv->auto_detected_encoding;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_newline_type:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: the detected newline type.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceNewlineType
Packit a7d494
gtk_source_file_loader_get_newline_type (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader),
Packit a7d494
			      GTK_SOURCE_NEWLINE_TYPE_LF);
Packit a7d494
Packit a7d494
	return loader->priv->auto_detected_newline_type;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_loader_get_compression_type:
Packit a7d494
 * @loader: a #GtkSourceFileLoader.
Packit a7d494
 *
Packit a7d494
 * Returns: the detected compression type.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceCompressionType
Packit a7d494
gtk_source_file_loader_get_compression_type (GtkSourceFileLoader *loader)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_LOADER (loader),
Packit a7d494
			      GTK_SOURCE_COMPRESSION_TYPE_NONE);
Packit a7d494
Packit a7d494
	return loader->priv->auto_detected_compression_type;
Packit a7d494
}