/* GIMP - The GNU Image Manipulation Program * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * libcurl backend for the URI plug-in * Copyright (C) 2006 Mukund Sivaraman * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "config.h" #include #include #include #include #include #include #include #include "uri-backend.h" #include "libgimp/stdplugins-intl.h" /* private variables */ static gchar *supported_protocols = NULL; static gchar *user_agent = NULL; /* public functions */ gboolean uri_backend_init (const gchar *plugin_name, gboolean run, GimpRunMode run_mode, GError **error) { GString *protocols; curl_version_info_data *vinfo; if (curl_global_init (CURL_GLOBAL_ALL)) { g_set_error (error, G_FILE_ERROR, G_FILE_ERROR_FAILED, "%s", _("Could not initialize libcurl")); return FALSE; } vinfo = curl_version_info (CURLVERSION_NOW); protocols = g_string_new ("http:,ftp:"); if (vinfo->features & CURL_VERSION_SSL) { g_string_append (protocols, ",https:,ftps:"); } supported_protocols = g_string_free (protocols, FALSE); user_agent = g_strconcat ("GIMP/", GIMP_VERSION, NULL); return TRUE; } void uri_backend_shutdown (void) { g_free (user_agent); g_free (supported_protocols); curl_global_cleanup (); } const gchar * uri_backend_get_load_help (void) { return "Loads a file using the libcurl file transfer library"; } const gchar * uri_backend_get_save_help (void) { return NULL; } const gchar * uri_backend_get_load_protocols (void) { return supported_protocols; } const gchar * uri_backend_get_save_protocols (void) { return NULL; } static int progress_callback (void *clientp, double dltotal, double dlnow, double ultotal, double ulnow) { gchar *memsize = NULL; if (dltotal > 0.0) { memsize = g_format_size_for_display (dltotal); gimp_progress_set_text_printf (_("Downloading %s of image data"), memsize); gimp_progress_update (dlnow / dltotal); } else { memsize = g_format_size_for_display (dlnow); gimp_progress_set_text_printf (_("Downloaded %s of image data"), memsize); gimp_progress_pulse (); } g_free (memsize); return 0; } gboolean uri_backend_load_image (const gchar *uri, const gchar *tmpname, GimpRunMode run_mode, GError **error) { FILE *out_file; CURL *curl_handle; CURLcode result; gint response_code; gimp_progress_init (_("Connecting to server")); if ((out_file = g_fopen (tmpname, "wb")) == NULL) { g_set_error (error, G_FILE_ERROR, g_file_error_from_errno (errno), _("Could not open '%s' for writing: %s"), gimp_filename_to_utf8 (tmpname), g_strerror (errno)); return FALSE; } curl_handle = curl_easy_init (); curl_easy_setopt (curl_handle, CURLOPT_URL, uri); curl_easy_setopt (curl_handle, CURLOPT_WRITEDATA, out_file); curl_easy_setopt (curl_handle, CURLOPT_PROGRESSFUNCTION, progress_callback); curl_easy_setopt (curl_handle, CURLOPT_NOPROGRESS, FALSE); curl_easy_setopt (curl_handle, CURLOPT_USERAGENT, user_agent); curl_easy_setopt (curl_handle, CURLOPT_FOLLOWLOCATION, TRUE); curl_easy_setopt (curl_handle, CURLOPT_MAXREDIRS, 10); curl_easy_setopt (curl_handle, CURLOPT_SSL_VERIFYPEER, FALSE); /* the following empty string causes libcurl to send a list of * all supported encodings which turns on compression * if libcurl has support for compression */ curl_easy_setopt (curl_handle, CURLOPT_ENCODING, ""); if ((result = curl_easy_perform (curl_handle)) != 0) { fclose (out_file); g_set_error (error, G_FILE_ERROR, G_FILE_ERROR_FAILED, _("Could not open '%s' for reading: %s"), uri, curl_easy_strerror (result)); curl_easy_cleanup (curl_handle); return FALSE; } curl_easy_getinfo (curl_handle, CURLINFO_RESPONSE_CODE, &response_code); if (response_code != 200) { fclose (out_file); g_set_error (error, G_FILE_ERROR, G_FILE_ERROR_FAILED, _("Opening '%s' for reading resulted in HTTP " "response code: %d"), uri, response_code); curl_easy_cleanup (curl_handle); return FALSE; } fclose (out_file); gimp_progress_update (1.0); curl_easy_cleanup (curl_handle); return TRUE; } gboolean uri_backend_save_image (const gchar *uri, const gchar *tmpname, GimpRunMode run_mode, GError **error) { g_set_error (error, G_FILE_ERROR, G_FILE_ERROR_FAILED, "not implemented"); return FALSE; } gchar * uri_backend_map_image (const gchar *uri, GimpRunMode run_mode) { return NULL; }