app-manager/app-manager.c
author J. Ali Harlow <ali@juiblex.co.uk>
Tue Jun 29 10:08:33 2021 +0100 (2021-06-29)
changeset 106 cc42fad3fe31
parent 103 c4b0d5cc34bc
child 109 2947214c450e
permissions -rw-r--r--
Add a --paths option to app-manager and use it in setup.js and update.js
     1 /*
     2  * Copyright (C) 2010, 2020  J. Ali Harlow <ali@juiblex.co.uk>
     3  *
     4  * This program is free software; you can redistribute it and/or modify
     5  * it under the terms of the GNU General Public License as published by
     6  * the Free Software Foundation; either version 2 of the License, or
     7  * (at your option) any later version.
     8  *
     9  * This program is distributed in the hope that it will be useful,
    10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    12  * GNU General Public License for more details.
    13  *
    14  * You should have received a copy of the GNU General Public License along
    15  * with this program; if not, write to the Free Software Foundation, Inc.,
    16  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
    17  */
    18 
    19 #include "config.h"
    20 #include <stdlib.h>
    21 #include <string.h>
    22 #ifdef WIN32
    23 #include <windows.h>
    24 #endif	/* WIN32 */
    25 #include <lua.h>
    26 #include <glib.h>
    27 #include <gio/gio.h>
    28 #include <gtk/gtk.h>
    29 #include <whelk/whelk.h>
    30 #include <plover/plover.h>
    31 #include <plover/packageset.h>
    32 #include <plover-gtk/stockicons.h>
    33 #include "app-manager.h"
    34 #include "localmedia.h"
    35 
    36 LUALIB_API int luaopen_posix(lua_State *L);
    37 
    38 #define LOGO_NAME	"plover-applications"
    39 
    40 GtkBuilder *ui;
    41 GtkTreeModel *installed,*applications,*location,*local_media;
    42 char *prefix=NULL;
    43 struct razor_relocations *relocations=NULL;
    44 
    45 void show_busy_cursor(gboolean busy)
    46 {
    47     GList *list,*link,*remaining;
    48     GdkDisplay *display;
    49     GdkCursor *cursor;
    50     GtkWidget *w;
    51     list=gtk_window_list_toplevels();
    52     while(list)
    53     {
    54 	w=GTK_WIDGET(list->data);
    55 	if (!w->window)
    56 	{
    57 	    link=list;
    58 	    list=g_list_remove_link(list,link);
    59 	    g_list_free_1(link);
    60 	}
    61 	else
    62 	{
    63 	    display=gtk_widget_get_display(w);
    64 	    cursor=busy?gdk_cursor_new_for_display(display,GDK_WATCH):NULL;
    65 	    remaining=NULL;
    66 	    for(link=list;link;link=link->next)
    67 	    {
    68 		w=GTK_WIDGET(link->data);
    69 		if (w->window)
    70 		{
    71 		    if (gtk_widget_get_display(w)==display)
    72 			gdk_window_set_cursor(w->window,cursor);
    73 		    else
    74 			remaining=g_list_prepend(remaining,w);
    75 		}
    76 	    }
    77 	    gdk_display_flush(display);
    78 	    if (cursor)
    79 		gdk_cursor_unref(cursor);
    80 	    g_list_free(list);
    81 	    list=remaining;
    82 	}
    83     }
    84 }
    85 
    86 /*
    87  * In Gtk+ 2.16.6, the default handler generates g_warnings on error.
    88  * It should display an error to the user. Do it ourselves.
    89  */
    90 
    91 static void show_uri(GtkLinkButton *button,const gchar *uri,gpointer data)
    92 {
    93     GdkScreen *screen;
    94     GtkWidget *dialog;
    95     GError *error=NULL;
    96     if (gtk_widget_has_screen(GTK_WIDGET(button)))
    97 	screen=gtk_widget_get_screen(GTK_WIDGET(button));
    98     else
    99 	screen=NULL;
   100     gtk_show_uri(screen,uri,GDK_CURRENT_TIME,&error);
   101     if (error)
   102     {
   103 	dialog=gtk_message_dialog_new(
   104 	  GTK_WINDOW(gtk_builder_get_object(ui,"MainWindow")),
   105 	  GTK_DIALOG_DESTROY_WITH_PARENT,GTK_MESSAGE_ERROR,GTK_BUTTONS_CLOSE,
   106 	  "Unable to show '%s'",uri);
   107 	gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog),
   108 	  error->message);
   109 	g_error_free(error);
   110 	gtk_dialog_run(GTK_DIALOG(dialog));
   111 	gtk_widget_destroy(dialog);
   112     }
   113 }
   114 
   115 #if 0
   116 /* Checks whether a loader for SVG files has been registered
   117  * with GdkPixbuf.
   118  */
   119 static gboolean pixbuf_supports_svg(void)
   120 {
   121     GSList *formats;
   122     GSList *tmp_list;
   123     static gint found_svg=-1;
   124     gchar **mime_types,**mime_type;
   125     if (found_svg!=-1)
   126 	return found_svg;
   127     formats=gdk_pixbuf_get_formats();
   128     found_svg=FALSE;
   129     for (tmp_list=formats;tmp_list && !found_svg;tmp_list=tmp_list->next)
   130     {
   131 	mime_types=gdk_pixbuf_format_get_mime_types(tmp_list->data);
   132 	for (mime_type=mime_types;*mime_type && !found_svg;mime_type++)
   133 	    if (!strcmp(*mime_type,"image/svg"))
   134 		found_svg=TRUE;
   135 	g_strfreev(mime_types);
   136     }
   137     g_slist_free(formats);
   138     return found_svg;
   139 }
   140 
   141 static void install_icon_at_size(const char *icon_name,GtkIconSet *icon_set,
   142   GtkIconSize size,const char *filename)
   143 {
   144     int w,h;
   145     GdkPixbuf *pixbuf;
   146     GtkIconSource *source;
   147     if (gtk_icon_size_lookup(size,&w,&h))
   148     {
   149 	pixbuf=gdk_pixbuf_new_from_file_at_size(filename,w,h,NULL);
   150 	if (pixbuf)
   151 	{
   152 	    source=gtk_icon_source_new();
   153 	    gtk_icon_source_set_size_wildcarded(source,FALSE);
   154 	    gtk_icon_source_set_size(source,size);
   155 	    gtk_icon_source_set_pixbuf(source,pixbuf);
   156 	    gtk_icon_set_add_source(icon_set,source);
   157 	    gtk_icon_source_free(source);
   158 	    g_object_unref(pixbuf);
   159 	}
   160     }
   161 }
   162 
   163 static void install_icons(void)
   164 {
   165     gchar *s;
   166     GtkIconSource *source;
   167     GtkIconSet *icon_set;
   168     GtkIconFactory *factory;
   169     factory=gtk_icon_factory_new();
   170     icon_set=gtk_icon_set_new();
   171     if (pixbuf_supports_svg())
   172     {
   173 	source=gtk_icon_source_new();
   174 	s=g_build_filename(prefix?prefix:"/usr",
   175 	  "share/icons/hicolor/scalable/apps/plover-applications.svg",NULL);
   176 	gtk_icon_source_set_filename(source,s);
   177 	g_free(s);
   178 	gtk_icon_set_add_source(icon_set,source);
   179 	gtk_icon_source_free(source);
   180     }
   181     else
   182     {
   183 	s=g_build_filename(prefix?prefix:"/usr",
   184 	  "share/icons/hicolor/24x24/apps/plover-applications.png",NULL);
   185 	install_icon_at_size(LOGO_NAME,icon_set,GTK_ICON_SIZE_MENU,s);
   186 	install_icon_at_size(LOGO_NAME,icon_set,GTK_ICON_SIZE_BUTTON,s);
   187 	install_icon_at_size(LOGO_NAME,icon_set,GTK_ICON_SIZE_SMALL_TOOLBAR,s);
   188 	install_icon_at_size(LOGO_NAME,icon_set,GTK_ICON_SIZE_LARGE_TOOLBAR,s);
   189 	g_free(s);
   190 	s=g_build_filename(prefix?prefix:"/usr",
   191 	  "share/icons/hicolor/48x48/apps/plover-applications.png",NULL);
   192 	install_icon_at_size(LOGO_NAME,icon_set,GTK_ICON_SIZE_DND,s);
   193 	install_icon_at_size(LOGO_NAME,icon_set,GTK_ICON_SIZE_DIALOG,s);
   194 	g_free(s);
   195     }
   196     gtk_icon_factory_add(factory,LOGO_NAME,icon_set);
   197     gtk_icon_set_unref(icon_set);
   198     icon_set=gtk_icon_factory_lookup(factory,LOGO_NAME);
   199     gtk_icon_factory_add_default(factory);
   200     g_object_unref(factory);
   201     icon_set=gtk_icon_factory_lookup_default(LOGO_NAME);
   202     gtk_window_set_default_icon_name(LOGO_NAME);
   203 }
   204 #endif
   205 
   206 static void install_icons(void)
   207 {
   208     GtkIconSet *icon_set;
   209     plover_icons_add_to_stock("apps",LOGO_NAME);
   210     icon_set=gtk_icon_factory_lookup_default(LOGO_NAME);
   211     gtk_window_set_default_icon_name(LOGO_NAME);
   212 }
   213 
   214 static gboolean uri_validate(const char *uri)
   215 {
   216     char *s;
   217     s=razor_path_relative_to_uri(uri,".",NULL);
   218     free(s);
   219     return !!s;
   220 }
   221 
   222 char *uri_from_base(const char *base,gboolean use_paths)
   223 {
   224     char *retval;
   225     if (use_paths)
   226     {
   227 	retval=razor_path_to_uri(base);
   228 	if (!retval)
   229 	{
   230 	    g_printerr("%s: Not a valid path\n",base);
   231 	    exit(1);
   232 	}
   233     }
   234     else if (!uri_validate(base))
   235     {
   236 	g_printerr("%s: Not a valid URI\n",base);
   237 	exit(1);
   238     }
   239     else
   240 	retval=strdup(base);
   241     return retval;
   242 }
   243 
   244 int main(int argc,char **argv)
   245 {
   246     gboolean use_paths=FALSE;
   247     GError *err=0;
   248     GtkWidget *w;
   249     char *uri;
   250     gchar *s,*database_uri,*contents;
   251     gchar *database=NULL,*setup_base=NULL,*update_base=NULL;
   252     gchar *default_action_base=NULL;
   253     gsize len;
   254     struct comps *comps;
   255     PloverPackageSet *set=NULL;
   256     GSList *objects,*lnk;
   257     gboolean started;
   258     GOptionEntry options[]={
   259 	{"database",0,0,G_OPTION_ARG_STRING,&database,
   260 	  "Operate on a distribution-local database","vendor/distribution"},
   261 	{"paths",0,0,G_OPTION_ARG_NONE,&use_paths,
   262 	  "Interpret locations as paths rather than URIs",NULL},
   263 	{"setup",0,0,G_OPTION_ARG_STRING,&setup_base,
   264 	  "Setup from installation media","location"},
   265 	{"update",0,0,G_OPTION_ARG_STRING,&update_base,
   266 	  "Update from upgrade media","location"},
   267 	{"default-action",0,0,G_OPTION_ARG_STRING,&default_action_base,
   268 	  "Install, remove or update from repository","location"},
   269 	{NULL}
   270     };
   271 #ifdef WIN32
   272     /*
   273      * app-manager is normally a GUI application, but rpm scripts may well
   274      * call console applications and it looks ugly if console windows keep
   275      * popping up. Avoid this by allocating our own console and hiding it.
   276      * Note:
   277      *	- If app-manager is a console application (typically for debugging),
   278      *    then skip this step.
   279      *  - Call ShowWindow twice to negate special handling on first call.
   280      */
   281     if (!GetConsoleWindow())
   282     {
   283 	AllocConsole();
   284 	ShowWindow(GetConsoleWindow(),SW_HIDE);
   285 	ShowWindow(GetConsoleWindow(),SW_HIDE);
   286     }
   287 #endif
   288     plover_exception_handler_init();
   289     razor_set_lua_loader("posix",(void (*)())luaopen_posix);
   290     razor_set_lua_loader("whelk",(void (*)())luaopen_whelk);
   291     if (!gtk_init_with_args(&argc,&argv,NULL,options,NULL,&err))
   292     {
   293 	g_printerr("%s\n",err->message);
   294 	exit(1);
   295     }
   296     if (!!setup_base+!!update_base+!!default_action_base>1)
   297     {
   298 	g_printerr(
   299 	  "--setup, --update and --default_action are mutually exclusive\n");
   300 	exit(1);
   301     }
   302 #ifdef WIN32
   303     prefix=g_win32_get_package_installation_directory_of_module(NULL);
   304 #endif
   305     install_icons();
   306     ui=gtk_builder_new();
   307     if (!g_file_get_contents("app-manager.ui",&contents,&len,&err) &&
   308       g_error_matches(err,G_FILE_ERROR,G_FILE_ERROR_NOENT))
   309     {
   310 #ifdef WIN32
   311 	s=g_build_filename(prefix,"share","plover","app-manager.ui",NULL);
   312 #else
   313 	s=g_build_filename(PLOVER_DATADIR,"app-manager.ui",NULL);
   314 #endif
   315 	g_clear_error(&err);
   316 	(void)g_file_get_contents(s,&contents,&len,&err);
   317 	g_free(s);
   318     }
   319     if (!err)
   320     {
   321 	(void)gtk_builder_add_from_string(ui,contents,len,&err);
   322 	g_free(contents);
   323     }
   324     if (err)
   325     {
   326 	g_error("%s",err->message);
   327 	exit(0);
   328     }
   329     gtk_builder_connect_signals(ui,NULL);
   330     gtk_link_button_set_uri_hook(show_uri,NULL,NULL);
   331     if (setup_base)
   332     {
   333 	uri=uri_from_base(setup_base,use_paths);
   334 	started=setup(uri);
   335 	free(uri);
   336     }
   337     else if (update_base)
   338     {
   339 	uri=uri_from_base(update_base,use_paths);
   340 	started=update(uri);
   341 	free(uri);
   342     }
   343     else if (default_action_base)
   344     {
   345 	uri=uri_from_base(default_action_base,use_paths);
   346 	started=default_action(uri);
   347 	free(uri);
   348     }
   349     else
   350     {
   351 	if (database)
   352 	{
   353 	    g_free(prefix);
   354 	    prefix=NULL;
   355 	    s=strchr(database,'/');
   356 	    if (*s)
   357 		*s++='\0';
   358 	    comps=plover_comps_new();
   359 	    plover_comps_set_vendor(comps,database);
   360 	    if (s)
   361 	    {
   362 		plover_comps_set_distribution(comps,s);
   363 		*--s='/';
   364 	    }
   365 	    prefix=plover_comps_get_default_prefix(comps);
   366 	    plover_comps_free(comps);
   367 	    s=g_strconcat(prefix,"/var/lib/razor",NULL);
   368 	    database_uri=razor_path_to_uri(s);
   369 	    g_free(s);
   370 	    razor_set_database_uri(database_uri);
   371 	    g_free(database_uri);
   372 	}
   373 	if (prefix)
   374 	{
   375 	    relocations=razor_relocations_create();
   376 	    razor_relocations_add(relocations,"/usr",prefix);
   377 	}
   378 	installed=GTK_TREE_MODEL(plover_package_store_new());
   379 	set=plover_package_set_new();
   380 	(void)plover_package_set_open(set,"",TRUE,NULL);
   381 	plover_package_store_add_set(PLOVER_PACKAGE_STORE(installed),set);
   382 	if (plover_package_set_get_no_details(set))
   383 	{
   384 	    w=GTK_WIDGET(gtk_builder_get_object(ui,"ViewFiles"));
   385 	    gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(w),TRUE);
   386 	}
   387 	applications=plover_applications_model_new(installed);
   388 	set_package_model(applications);
   389 	w=GTK_WIDGET(gtk_builder_get_object(ui,"MainWindow"));
   390 	gtk_widget_show(w);
   391 	started=TRUE;
   392     }
   393     if (started)
   394 	gtk_main();
   395     g_clear_object(&set);
   396     objects=gtk_builder_get_objects(ui);
   397     for(lnk=objects;lnk;lnk=lnk->next)
   398 	if (GTK_IS_WIDGET(lnk->data) &&
   399 	  gtk_widget_is_toplevel(GTK_WIDGET(lnk->data)))
   400 	    gtk_widget_destroy(GTK_WIDGET(lnk->data));
   401     g_slist_free(objects);
   402     g_clear_object(&ui);
   403     g_clear_object(&installed);
   404     g_clear_object(&applications);
   405     g_clear_object(&location);
   406     g_clear_object(&local_media);
   407     if (relocations)
   408 	razor_relocations_destroy(relocations);
   409     g_free(prefix);
   410     g_free(setup_base);
   411     g_free(update_base);
   412     g_free(default_action_base);
   413     g_free(database);
   414     exit(0);
   415 }
   416 
   417 G_MODULE_EXPORT void
   418   on_applications_toggled(GtkToggleToolButton *button,gpointer data)
   419 {
   420     if (gtk_toggle_tool_button_get_active(button))
   421     {
   422 	if (!applications)
   423 	    applications=plover_applications_model_new(installed);
   424 	set_package_model(applications);
   425     }
   426 }
   427 
   428 G_MODULE_EXPORT void
   429   on_all_packages_toggled(GtkToggleToolButton *button,gpointer data)
   430 {
   431     if (gtk_toggle_tool_button_get_active(button))
   432 	set_package_model(installed);
   433 }
   434 
   435 G_MODULE_EXPORT void
   436   on_local_media_toggled(GtkToggleToolButton *button,gpointer data)
   437 {
   438     if (gtk_toggle_tool_button_get_active(button))
   439     {
   440 	if (!local_media)
   441 	{
   442 	    show_busy_cursor(TRUE);
   443 	    local_media=plover_local_media_store_new();
   444 	    show_busy_cursor(FALSE);
   445 	}
   446 	set_package_model(local_media);
   447     }
   448 }
   449 
   450 G_MODULE_EXPORT void
   451   on_location_toggled(GtkToggleToolButton *button,gpointer data)
   452 {
   453     if (gtk_toggle_tool_button_get_active(button))
   454 	set_package_model(location);
   455 }
   456 
   457 G_MODULE_EXPORT void on_open_location(GtkWidget *widget)
   458 {
   459     GtkWidget *w=GTK_WIDGET(gtk_builder_get_object(ui,"MainWindow"));
   460     GtkWidget *dialog;
   461     GFile *file,*parent;
   462     GFileInfo *fi;
   463     GMount *mount;
   464     gchar *path,*name;
   465     PloverPackageSet *set;
   466     GSList *sets;
   467     GError *err=NULL;
   468     dialog=gtk_file_chooser_dialog_new("Open Location",GTK_WINDOW(w),
   469       GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,GTK_STOCK_CANCEL,
   470       GTK_RESPONSE_CANCEL,GTK_STOCK_OPEN,GTK_RESPONSE_ACCEPT,NULL);
   471 #if GTK_CHECK_VERSION(2,18,0)
   472     gtk_file_chooser_set_create_folders(GTK_FILE_CHOOSER(dialog),FALSE);
   473 #endif
   474     if (gtk_dialog_run(GTK_DIALOG(dialog))==GTK_RESPONSE_ACCEPT)
   475     {
   476 	show_busy_cursor(TRUE);
   477 	path=gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
   478 	set=plover_package_set_new_from_yum(path,relocations,&err);
   479 	if (set)
   480 	{
   481 	    if (!location)
   482 		location=GTK_TREE_MODEL(plover_package_store_new());
   483 	    while((sets=
   484 	      plover_package_store_get_sets(PLOVER_PACKAGE_STORE(location))))
   485 		plover_package_store_remove_set(PLOVER_PACKAGE_STORE(location),
   486 		  PLOVER_PACKAGE_SET(sets->data));
   487 	    plover_package_store_add_set(PLOVER_PACKAGE_STORE(location),set);
   488 	    g_object_unref(set);
   489 	    w=GTK_WIDGET(gtk_builder_get_object(ui,"LocationButton"));
   490 	    file=g_file_new_for_path(path);
   491 	    parent=g_file_get_parent(file);
   492 	    if (parent)
   493 	    {
   494 		g_object_unref(parent);
   495 		mount=NULL;
   496 	    }
   497 	    else
   498 		mount=g_file_find_enclosing_mount(file,NULL,NULL);
   499 	    if (mount)
   500 	    {
   501 		name=g_mount_get_name(mount);
   502 		g_object_unref(mount);
   503 	    }
   504 	    else
   505 	    {
   506 		fi=g_file_query_info(file,
   507 		  G_FILE_ATTRIBUTE_STANDARD_DISPLAY_NAME,
   508 		  G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,NULL,NULL);
   509 		if (fi)
   510 		{
   511 		    name=g_strdup(g_file_info_get_display_name(fi));
   512 		    g_object_unref(fi);
   513 		}
   514 		else
   515 		    name=g_filename_display_basename(path);
   516 		g_object_unref(file);
   517 	    }
   518 	    gtk_tool_button_set_label(GTK_TOOL_BUTTON(w),name);
   519 	    g_free(name);
   520 	    gtk_widget_show(w);
   521 	    gtk_toggle_tool_button_set_active(GTK_TOGGLE_TOOL_BUTTON(w),TRUE);
   522 	}
   523 	else
   524 	{
   525 	    gtk_widget_destroy(dialog);
   526 	    dialog=gtk_message_dialog_new(GTK_WINDOW(w),
   527 	      GTK_DIALOG_DESTROY_WITH_PARENT,GTK_MESSAGE_ERROR,
   528 	      GTK_BUTTONS_CLOSE,"Error loading repository '%s'",path);
   529 	    gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(dialog),
   530 	      "%s",err->message);
   531 	    gtk_dialog_run(GTK_DIALOG(dialog));
   532 	    g_error_free(err);
   533 	}
   534 	g_free(path);
   535 	show_busy_cursor(FALSE);
   536     }
   537     gtk_widget_destroy(dialog);
   538 }
   539 
   540 G_MODULE_EXPORT void on_scan_local_media(GtkWidget *widget)
   541 {
   542     show_busy_cursor(TRUE);
   543     if (!local_media)
   544 	local_media=plover_local_media_store_new();
   545     plover_local_media_store_scan(PLOVER_LOCAL_MEDIA_STORE(local_media));
   546     show_busy_cursor(FALSE);
   547 }
   548 
   549 G_MODULE_EXPORT void on_help_about(GtkWidget *widget)
   550 {
   551     GtkWidget *w=GTK_WIDGET(gtk_builder_get_object(ui,"MainWindow"));
   552     gtk_show_about_dialog(GTK_WINDOW(w),"name",PACKAGE_NAME,
   553       "version",PACKAGE_VERSION,"comments","Application Manager",
   554       "copyright","Copyright © 2010, 2014 J. Ali Harlow",
   555       "logo-icon-name",LOGO_NAME,
   556       NULL);
   557 }
   558 
   559 G_MODULE_EXPORT void on_find_clicked(GtkButton *button)
   560 {
   561     gchar *text;
   562     GtkWidget *w=GTK_WIDGET(gtk_builder_get_object(ui,"SearchEntry"));
   563     text=g_strdup(gtk_entry_get_text(GTK_ENTRY(w)));
   564     gtk_entry_set_text(GTK_ENTRY(w),"");
   565     gtk_entry_set_text(GTK_ENTRY(w),text);
   566     g_free(text);
   567 }