/* * text-highlight.c * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, see . * */ #ifdef HAVE_CONFIG_H #include #endif #include "e-mail-formatter-text-highlight.h" #include "languages.h" /* FIXME Delete these once we can use GSubprocess. */ #include #include #include #include #include #include #include #include #include #include #include typedef EMailFormatterExtension EMailFormatterTextHighlight; typedef EMailFormatterExtensionClass EMailFormatterTextHighlightClass; typedef EExtension EMailFormatterTextHighlightLoader; typedef EExtensionClass EMailFormatterTextHighlightLoaderClass; typedef struct _TextHighlightClosure TextHighlightClosure; struct _TextHighlightClosure { GMainLoop *main_loop; GError *input_error; GError *output_error; }; GType e_mail_formatter_text_highlight_get_type (void); G_DEFINE_DYNAMIC_TYPE ( EMailFormatterTextHighlight, e_mail_formatter_text_highlight, E_TYPE_MAIL_FORMATTER_EXTENSION) static gchar * get_syntax (EMailPart *part, const gchar *uri) { gchar *syntax = NULL; CamelContentType *ct = NULL; CamelMimePart *mime_part; mime_part = e_mail_part_ref_mime_part (part); if (uri) { SoupURI *soup_uri = soup_uri_new (uri); GHashTable *query = soup_form_decode (soup_uri->query); syntax = g_hash_table_lookup (query, "__formatas"); if (syntax) { syntax = g_strdup (syntax); } g_hash_table_destroy (query); soup_uri_free (soup_uri); } /* Try to detect syntax by content-type first */ if (syntax == NULL) { ct = camel_mime_part_get_content_type (mime_part); if (ct) { gchar *mime_type = camel_content_type_simple (ct); syntax = (gchar *) get_syntax_for_mime_type (mime_type); syntax = syntax ? g_strdup (syntax) : NULL; g_free (mime_type); } } /* If it fails or the content type too generic, try to detect it by * filename extension */ if (syntax == NULL || (ct != NULL && (camel_content_type_is (ct, "application", "octet-stream") || (camel_content_type_is (ct, "text", "plain"))))) { const gchar *filename; filename = camel_mime_part_get_filename (mime_part); if (filename != NULL) { gchar *ext = g_strrstr (filename, "."); if (ext != NULL) { g_free (syntax); syntax = (gchar *) get_syntax_for_ext (ext + 1); syntax = syntax ? g_strdup (syntax) : NULL; } } } /* Out of ideas - use plain text */ if (syntax == NULL) { syntax = g_strdup ("txt"); } g_object_unref (mime_part); return syntax; } static void text_highlight_input_spliced (GObject *source_object, GAsyncResult *result, gpointer user_data) { TextHighlightClosure *closure = user_data; g_output_stream_splice_finish ( G_OUTPUT_STREAM (source_object), result, &closure->input_error); } static void text_highlight_output_spliced (GObject *source_object, GAsyncResult *result, gpointer user_data) { TextHighlightClosure *closure = user_data; g_output_stream_splice_finish ( G_OUTPUT_STREAM (source_object), result, &closure->output_error); g_main_loop_quit (closure->main_loop); } static gboolean text_highlight_feed_data (GOutputStream *output_stream, CamelDataWrapper *data_wrapper, gint pipe_stdin, gint pipe_stdout, GCancellable *cancellable, GError **error) { TextHighlightClosure closure; GInputStream *input_stream = NULL; GOutputStream *temp_stream = NULL; GInputStream *stdout_stream = NULL; GOutputStream *stdin_stream = NULL; GMainContext *main_context; gchar *utf8_data; gconstpointer data; gsize size; gboolean success; /* We need to dump CamelDataWrapper to a buffer, force the content * to valid UTF-8, feed the UTF-8 data to the 'highlight' process, * read the converted data back and feed it to the CamelStream. */ /* FIXME Use GSubprocess once we can require GLib 2.40. */ temp_stream = g_memory_output_stream_new_resizable (); success = camel_data_wrapper_decode_to_output_stream_sync ( data_wrapper, temp_stream, cancellable, error); if (!success) goto exit; main_context = g_main_context_new (); closure.main_loop = g_main_loop_new (main_context, FALSE); closure.input_error = NULL; closure.output_error = NULL; g_main_context_push_thread_default (main_context); data = g_memory_output_stream_get_data ( G_MEMORY_OUTPUT_STREAM (temp_stream)); size = g_memory_output_stream_get_data_size ( G_MEMORY_OUTPUT_STREAM (temp_stream)); /* FIXME Write a GConverter that does this so we can decode * straight to the stdin pipe and skip all this extra * buffering. */ utf8_data = e_util_utf8_data_make_valid ((gchar *) data, size); g_clear_object (&temp_stream); /* Takes ownership of the UTF-8 string. */ input_stream = g_memory_input_stream_new_from_data ( utf8_data, -1, (GDestroyNotify) g_free); stdin_stream = g_unix_output_stream_new (pipe_stdin, TRUE); stdout_stream = g_unix_input_stream_new (pipe_stdout, TRUE); /* Splice the streams together. */ /* GCancellable is only supposed to be used in one operation * at a time. Skip it here and use it for reading converted * data, since that operation terminates the main loop. */ g_output_stream_splice_async ( stdin_stream, input_stream, G_OUTPUT_STREAM_SPLICE_CLOSE_SOURCE | G_OUTPUT_STREAM_SPLICE_CLOSE_TARGET, G_PRIORITY_DEFAULT, NULL, text_highlight_input_spliced, &closure); g_output_stream_splice_async ( output_stream, stdout_stream, G_OUTPUT_STREAM_SPLICE_CLOSE_SOURCE | G_OUTPUT_STREAM_SPLICE_CLOSE_TARGET, G_PRIORITY_DEFAULT, cancellable, text_highlight_output_spliced, &closure); g_main_loop_run (closure.main_loop); g_main_context_pop_thread_default (main_context); g_main_context_unref (main_context); g_main_loop_unref (closure.main_loop); g_clear_object (&input_stream); g_clear_object (&stdin_stream); g_clear_object (&stdout_stream); if (closure.input_error != NULL) { g_propagate_error (error, closure.input_error); g_clear_error (&closure.output_error); success = FALSE; goto exit; } if (closure.output_error != NULL) { g_propagate_error (error, closure.output_error); success = FALSE; goto exit; } exit: g_clear_object (&temp_stream); return success; } static gboolean emfe_text_highlight_format (EMailFormatterExtension *extension, EMailFormatter *formatter, EMailFormatterContext *context, EMailPart *part, GOutputStream *stream, GCancellable *cancellable) { CamelMimePart *mime_part; CamelContentType *ct; gboolean success = FALSE; mime_part = e_mail_part_ref_mime_part (part); ct = camel_mime_part_get_content_type (mime_part); /* Don't format text/html unless it's an attachment */ if (ct && camel_content_type_is (ct, "text", "html")) { const CamelContentDisposition *disp; disp = camel_mime_part_get_content_disposition (mime_part); if (disp == NULL) goto exit; if (g_strcmp0 (disp->disposition, "attachment") != 0) goto exit; } if (context->mode == E_MAIL_FORMATTER_MODE_PRINTING) { /* Don't interfere with printing. */ goto exit; } else if (context->mode == E_MAIL_FORMATTER_MODE_RAW) { gint pipe_stdin, pipe_stdout; GPid pid; CamelDataWrapper *dw; gchar *font_family, *font_size, *syntax; PangoFontDescription *fd; GSettings *settings; gchar *font = NULL; const gchar *argv[] = { HIGHLIGHT_COMMAND, NULL, /* --font= */ NULL, /* --font-size= */ NULL, /* --syntax= */ "--out-format=html", "--include-style", "--inline-css", "--style=bclear", "--failsafe", NULL }; dw = camel_medium_get_content (CAMEL_MEDIUM (mime_part)); if (dw == NULL) goto exit; syntax = get_syntax (part, context->uri); /* Use the traditional text/plain formatter for plain-text */ if (g_strcmp0 (syntax, "txt") == 0) { g_free (syntax); goto exit; } settings = g_settings_new ("org.gnome.evolution.mail"); if (g_settings_get_boolean (settings, "use-custom-font")) font = g_settings_get_string ( settings, "monospace-font"); g_object_unref (settings); if (font == NULL) { settings = g_settings_new ( "org.gnome.desktop.interface"); font = g_settings_get_string ( settings, "monospace-font-name"); g_object_unref (settings); } if (font == NULL) font = g_strdup ("monospace 10"); fd = pango_font_description_from_string (font); g_free (font); font_family = g_strdup_printf ( "--font='%s'", pango_font_description_get_family (fd)); font_size = g_strdup_printf ( "--font-size=%d", pango_font_description_get_size (fd) / PANGO_SCALE); argv[1] = font_family; argv[2] = font_size; argv[3] = g_strdup_printf ("--syntax=%s", syntax); g_free (syntax); success = g_spawn_async_with_pipes ( NULL, (gchar **) argv, NULL, 0, NULL, NULL, &pid, &pipe_stdin, &pipe_stdout, NULL, NULL); if (success) { GError *local_error = NULL; success = text_highlight_feed_data ( stream, dw, pipe_stdin, pipe_stdout, cancellable, &local_error); if (g_error_matches ( local_error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) { /* Do nothing. */ } else if (local_error != NULL) { g_warning ( "%s: %s", G_STRFUNC, local_error->message); } g_clear_error (&local_error); g_spawn_close_pid (pid); } if (!success) { /* We can't call e_mail_formatter_format_as on text/plain, * because text-highlight is registered as an handler for * text/plain, so we would end up in an endless recursion. * * Just return FALSE here and EMailFormatter will automatically * fall back to the default text/plain formatter */ if (camel_content_type_is (ct, "text", "plain")) { g_free (font_family); g_free (font_size); g_free ((gchar *) argv[3]); pango_font_description_free (fd); goto exit; } else { /* In case of any other content, force use of * text/plain formatter, because returning FALSE * for text/x-patch or application/php would show * an error, as there is no other handler registered * for these */ e_mail_formatter_format_as ( formatter, context, part, stream, "application/vnd.evolution.plaintext", cancellable); } } g_free (font_family); g_free (font_size); g_free ((gchar *) argv[3]); pango_font_description_free (fd); } else { CamelFolder *folder; const gchar *message_uid; const gchar *default_charset, *charset; gchar *uri, *str; gchar *syntax; folder = e_mail_part_list_get_folder (context->part_list); message_uid = e_mail_part_list_get_message_uid (context->part_list); default_charset = e_mail_formatter_get_default_charset (formatter); charset = e_mail_formatter_get_charset (formatter); if (!default_charset) default_charset = ""; if (!charset) charset = ""; syntax = get_syntax (part, NULL); uri = e_mail_part_build_uri ( folder, message_uid, "part_id", G_TYPE_STRING, e_mail_part_get_id (part), "mode", G_TYPE_INT, E_MAIL_FORMATTER_MODE_RAW, "__formatas", G_TYPE_STRING, syntax, "formatter_default_charset", G_TYPE_STRING, default_charset, "formatter_charset", G_TYPE_STRING, charset, NULL); g_free (syntax); str = g_strdup_printf ( "
" "" "
", e_mail_part_get_id (part), e_mail_part_get_id (part), uri); g_output_stream_write_all ( stream, str, strlen (str), NULL, cancellable, NULL); g_free (str); g_free (uri); } success = TRUE; exit: g_object_unref (mime_part); return success; } static void e_mail_formatter_text_highlight_class_init (EMailFormatterExtensionClass *class) { class->display_name = _("Text Highlight"); class->description = _("Syntax highlighting of mail parts"); class->mime_types = get_mime_types (); class->format = emfe_text_highlight_format; } static void e_mail_formatter_text_highlight_class_finalize (EMailFormatterExtensionClass *class) { } static void e_mail_formatter_text_highlight_init (EMailFormatterExtension *extension) { } void e_mail_formatter_text_highlight_type_register (GTypeModule *type_module) { e_mail_formatter_text_highlight_register_type (type_module); }