482 lines
14 KiB
C
482 lines
14 KiB
C
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#ifdef HAVE_SYS_RESOURCE_H
|
|
#include <sys/time.h>
|
|
#include <sys/resource.h>
|
|
#endif
|
|
|
|
#include <Eina.h>
|
|
#include <Eet.h>
|
|
#include <Ecore.h>
|
|
#include <Ecore_File.h>
|
|
|
|
#define EFREET_MODULE_LOG_DOM _efreet_desktop_cache_log_dom
|
|
static int _efreet_desktop_cache_log_dom = -1;
|
|
|
|
#include "Efreet.h"
|
|
#include "efreet_private.h"
|
|
#include "efreet_cache_private.h"
|
|
|
|
static Eet_Data_Descriptor *edd = NULL;
|
|
static Eet_File *ef = NULL;
|
|
static Eet_File *util_ef = NULL;
|
|
|
|
static Eina_Hash *desktops = NULL;
|
|
|
|
static Eina_Hash *file_ids = NULL;
|
|
static Efreet_Cache_Hash *old_file_ids = NULL;
|
|
static Eina_Hash *paths = NULL;
|
|
|
|
static Eina_Hash *mime_types = NULL;
|
|
static Eina_Hash *categories = NULL;
|
|
static Eina_Hash *startup_wm_class = NULL;
|
|
static Eina_Hash *name = NULL;
|
|
static Eina_Hash *generic_name = NULL;
|
|
static Eina_Hash *comment = NULL;
|
|
static Eina_Hash *exec = NULL;
|
|
|
|
static int
|
|
cache_add(const char *path, const char *file_id, int priority EINA_UNUSED, int *changed)
|
|
{
|
|
Efreet_Desktop *desk;
|
|
char *ext;
|
|
|
|
INF("FOUND: %s", path);
|
|
if (file_id) INF(" (id): %s", file_id);
|
|
ext = strrchr(path, '.');
|
|
if (!ext || (strcmp(ext, ".desktop") && strcmp(ext, ".directory"))) return 1;
|
|
desk = efreet_desktop_new(path);
|
|
if (desk) INF(" OK");
|
|
else INF(" FAIL");
|
|
if (!desk) return 1;
|
|
if (!desk->eet)
|
|
{
|
|
/* This file isn't in cache */
|
|
*changed = 1;
|
|
INF(" NEW");
|
|
}
|
|
else if (ecore_file_mod_time(desk->orig_path) != desk->load_time)
|
|
{
|
|
efreet_desktop_free(desk);
|
|
*changed = 1;
|
|
desk = efreet_desktop_uncached_new(path);
|
|
if (desk) INF(" CHANGED");
|
|
else INF(" NO UNCACHED");
|
|
}
|
|
if (!desk) return 1;
|
|
if (file_id && old_file_ids && !eina_hash_find(old_file_ids->hash, file_id))
|
|
{
|
|
*changed = 1;
|
|
INF(" NOT IN UTILS");
|
|
}
|
|
if (!eina_hash_find(paths, desk->orig_path))
|
|
{
|
|
if (!eet_data_write(ef, edd, desk->orig_path, desk, 0))
|
|
return 0;
|
|
eina_hash_add(paths, desk->orig_path, (void *)1);
|
|
}
|
|
/* TODO: We should check priority, and not just hope we search in right order */
|
|
/* TODO: We need to find out if prioritized file id has changed because of
|
|
* changed search order. */
|
|
if (!desk->hidden && desk->type == EFREET_DESKTOP_TYPE_APPLICATION &&
|
|
file_id && !eina_hash_find(file_ids, file_id))
|
|
{
|
|
Eina_List *l;
|
|
char *data;
|
|
Efreet_Cache_Array_String *array;
|
|
|
|
#define ADD_LIST(list, hash) \
|
|
EINA_LIST_FOREACH((list), l, data) \
|
|
{ \
|
|
array = eina_hash_find((hash), data); \
|
|
if (!array) \
|
|
array = NEW(Efreet_Cache_Array_String, 1); \
|
|
array->array = realloc(array->array, sizeof (char *) * (array->array_count + 1)); \
|
|
array->array[array->array_count++] = desk->orig_path; \
|
|
eina_hash_set((hash), data, array); \
|
|
}
|
|
#define ADD_ELEM(elem, hash) \
|
|
if ((elem)) \
|
|
{ \
|
|
data = (elem); \
|
|
array = eina_hash_find((hash), data); \
|
|
if (!array) \
|
|
array = NEW(Efreet_Cache_Array_String, 1); \
|
|
array->array = realloc(array->array, sizeof (char *) * (array->array_count + 1)); \
|
|
array->array[array->array_count++] = desk->orig_path; \
|
|
eina_hash_set((hash), data, array); \
|
|
}
|
|
ADD_LIST(desk->mime_types, mime_types);
|
|
ADD_LIST(desk->categories, categories);
|
|
ADD_ELEM(desk->startup_wm_class, startup_wm_class);
|
|
ADD_ELEM(desk->name, name);
|
|
ADD_ELEM(desk->generic_name, generic_name);
|
|
ADD_ELEM(desk->comment, comment);
|
|
ADD_ELEM(desk->exec, exec);
|
|
eina_hash_add(file_ids, file_id, desk->orig_path);
|
|
eina_hash_add(desktops, desk->orig_path, desk);
|
|
}
|
|
else
|
|
efreet_desktop_free(desk);
|
|
return 1;
|
|
}
|
|
|
|
|
|
static int
|
|
cache_scan(const char *path, const char *base_id, int priority, int recurse, int *changed)
|
|
{
|
|
char *file_id = NULL;
|
|
char id[PATH_MAX];
|
|
char buf[PATH_MAX];
|
|
Eina_Iterator *it;
|
|
Eina_File_Direct_Info *info;
|
|
|
|
if (!ecore_file_is_dir(path)) return 1;
|
|
|
|
it = eina_file_stat_ls(path);
|
|
if (!it) return 1;
|
|
|
|
id[0] = '\0';
|
|
EINA_ITERATOR_FOREACH(it, info)
|
|
{
|
|
const char *fname;
|
|
|
|
fname = info->path + info->name_start;
|
|
if (base_id)
|
|
{
|
|
if (*base_id)
|
|
snprintf(id, sizeof(id), "%s-%s", base_id, fname);
|
|
else
|
|
{
|
|
strncpy(id, fname, PATH_MAX);
|
|
id[PATH_MAX - 1] = '\0';
|
|
}
|
|
file_id = id;
|
|
}
|
|
|
|
snprintf(buf, sizeof(buf), "%s/%s", path, fname);
|
|
if (info->type == EINA_FILE_DIR)
|
|
{
|
|
if (recurse)
|
|
cache_scan(buf, file_id, priority, recurse, changed);
|
|
}
|
|
else
|
|
{
|
|
if (!cache_add(buf, file_id, priority, changed))
|
|
{
|
|
eina_iterator_free(it);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
eina_iterator_free(it);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
cache_lock_file(void)
|
|
{
|
|
char file[PATH_MAX];
|
|
struct flock fl;
|
|
int lockfd;
|
|
|
|
snprintf(file, sizeof(file), "%s/efreet/desktop_data.lock", efreet_cache_home_get());
|
|
lockfd = open(file, O_CREAT | O_RDWR, S_IRUSR | S_IWUSR);
|
|
if (lockfd < 0) return -1;
|
|
efreet_fsetowner(lockfd);
|
|
|
|
memset(&fl, 0, sizeof(struct flock));
|
|
fl.l_type = F_WRLCK;
|
|
fl.l_whence = SEEK_SET;
|
|
if (fcntl(lockfd, F_SETLK, &fl) < 0)
|
|
{
|
|
INF("LOCKED! You may want to delete %s if this persists", file);
|
|
close(lockfd);
|
|
return -1;
|
|
}
|
|
|
|
return lockfd;
|
|
}
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
/* TODO:
|
|
* - Add file monitor on files, so that we catch changes on files
|
|
* during whilst this program runs.
|
|
* - Maybe linger for a while to reduce number of cache re-creates.
|
|
*/
|
|
Efreet_Cache_Hash hash;
|
|
Efreet_Cache_Version version;
|
|
Eina_List *dirs = NULL;
|
|
Eina_List *systemdirs = NULL;
|
|
Eina_List *extra_dirs = NULL;
|
|
Eina_List *l = NULL;
|
|
int priority = 0;
|
|
char *dir = NULL;
|
|
char *path;
|
|
int lockfd = -1, tmpfd;
|
|
int changed = 0;
|
|
int i;
|
|
char path_name[PATH_MAX] = { '\0' };
|
|
Eina_Tmpstr *file = NULL;
|
|
Eina_Tmpstr *util_file = NULL;
|
|
|
|
if (!eina_init()) goto eina_error;
|
|
_efreet_desktop_cache_log_dom =
|
|
eina_log_domain_register("efreet_desktop_cache", EFREET_DEFAULT_LOG_COLOR);
|
|
if (_efreet_desktop_cache_log_dom < 0)
|
|
{
|
|
EINA_LOG_ERR("Efreet: Could not create a log domain for efreet_desktop_cache.");
|
|
return -1;
|
|
}
|
|
|
|
for (i = 1; i < argc; i++)
|
|
{
|
|
if (!strcmp(argv[i], "-v"))
|
|
eina_log_domain_level_set("efreet_desktop_cache", EINA_LOG_LEVEL_DBG);
|
|
else if ((!strcmp(argv[i], "-h")) ||
|
|
(!strcmp(argv[i], "-help")) ||
|
|
(!strcmp(argv[i], "--h")) ||
|
|
(!strcmp(argv[i], "--help")))
|
|
{
|
|
printf("Options:\n");
|
|
printf(" -v Verbose mode\n");
|
|
printf(" -d dir1 dir2 Extra dirs\n");
|
|
exit(0);
|
|
}
|
|
else if (!strcmp(argv[i], "-d"))
|
|
{
|
|
while ((i < (argc - 1)) && (argv[(i + 1)][0] != '-'))
|
|
extra_dirs = eina_list_append(extra_dirs, argv[++i]);
|
|
}
|
|
}
|
|
extra_dirs = eina_list_sort(extra_dirs, -1, EINA_COMPARE_CB(strcmp));
|
|
|
|
#ifdef HAVE_SYS_RESOURCE_H
|
|
setpriority(PRIO_PROCESS, 0, 19);
|
|
#elif _WIN32
|
|
SetPriorityClass(GetCurrentProcess(), IDLE_PRIORITY_CLASS);
|
|
#endif
|
|
|
|
/* init external subsystems */
|
|
if (!eet_init()) goto eet_error;
|
|
if (!ecore_init()) goto ecore_error;
|
|
|
|
efreet_cache_update = 0;
|
|
/* finish efreet init */
|
|
if (!efreet_init()) goto efreet_error;
|
|
|
|
/* create homedir */
|
|
snprintf(path_name, sizeof(path_name), "%s/efreet", efreet_cache_home_get());
|
|
if (!ecore_file_exists(path_name))
|
|
{
|
|
if (!ecore_file_mkpath(path_name)) goto efreet_error;
|
|
efreet_setowner(path_name);
|
|
}
|
|
|
|
/* lock process, so that we only run one copy of this program */
|
|
lockfd = cache_lock_file();
|
|
if (lockfd == -1) goto efreet_error;
|
|
|
|
edd = efreet_desktop_edd();
|
|
if (!edd) goto edd_error;
|
|
|
|
/* read user dirs from old cache */
|
|
ef = eet_open(efreet_desktop_util_cache_file(), EET_FILE_MODE_READ);
|
|
if (ef)
|
|
{
|
|
old_file_ids = eet_data_read(ef, efreet_hash_string_edd(), "file_id");
|
|
eet_close(ef);
|
|
}
|
|
|
|
/* create cache */
|
|
tmpfd = eina_file_mkstemp("efreet_desktop_cache_XXXXXX", &file);
|
|
if (tmpfd < 0) goto error;
|
|
close(tmpfd);
|
|
ef = eet_open(file, EET_FILE_MODE_READ_WRITE);
|
|
if (!ef) goto error;
|
|
|
|
tmpfd = eina_file_mkstemp("efreet_desktop_util_cache_XXXXXX", &util_file);
|
|
if (tmpfd < 0) goto error;
|
|
close(tmpfd);
|
|
util_ef = eet_open(util_file, EET_FILE_MODE_READ_WRITE);
|
|
if (!util_ef) goto error;
|
|
|
|
/* write cache version */
|
|
version.major = EFREET_DESKTOP_UTILS_CACHE_MAJOR;
|
|
version.minor = EFREET_DESKTOP_UTILS_CACHE_MINOR;
|
|
eet_data_write(util_ef, efreet_version_edd(), EFREET_CACHE_VERSION, &version, 1);
|
|
version.major = EFREET_DESKTOP_CACHE_MAJOR;
|
|
version.minor = EFREET_DESKTOP_CACHE_MINOR;
|
|
eet_data_write(ef, efreet_version_edd(), EFREET_CACHE_VERSION, &version, 1);
|
|
|
|
desktops = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_desktop_free));
|
|
|
|
file_ids = eina_hash_string_superfast_new(NULL);
|
|
paths = eina_hash_string_superfast_new(NULL);
|
|
|
|
mime_types = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
categories = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
startup_wm_class = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
name = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
generic_name = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
comment = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
exec = eina_hash_string_superfast_new(EINA_FREE_CB(efreet_cache_array_string_free));
|
|
|
|
dirs = efreet_default_dirs_get(efreet_data_home_get(), efreet_data_dirs_get(),
|
|
"applications");
|
|
if (!dirs) goto error;
|
|
|
|
EINA_LIST_FREE(dirs, path)
|
|
{
|
|
char file_id[PATH_MAX] = { '\0' };
|
|
|
|
if (!cache_scan(path, file_id, priority++, 1, &changed)) goto error;
|
|
systemdirs = eina_list_append(systemdirs, path);
|
|
}
|
|
|
|
EINA_LIST_FOREACH(extra_dirs, l, path)
|
|
if (!cache_scan(path, NULL, priority, 0, &changed)) goto error;
|
|
|
|
/* store util */
|
|
#define STORE_HASH_ARRAY(_hash) \
|
|
if (eina_hash_population((_hash)) > 0) \
|
|
{ \
|
|
Eina_Iterator *it; \
|
|
Efreet_Cache_Array_String array; \
|
|
const char *str; \
|
|
\
|
|
hash.hash = (_hash); \
|
|
eet_data_write(util_ef, efreet_hash_array_string_edd(), #_hash "_hash", &hash, 1); \
|
|
array.array_count = 0; \
|
|
array.array = malloc(eina_hash_population(hash.hash) * sizeof(char *)); \
|
|
it = eina_hash_iterator_key_new(hash.hash); \
|
|
EINA_ITERATOR_FOREACH(it, str) \
|
|
array.array[array.array_count++] = str; \
|
|
eina_iterator_free(it); \
|
|
eet_data_write(util_ef, efreet_array_string_edd(), #_hash "_list", &array, 1); \
|
|
free(array.array); \
|
|
}
|
|
STORE_HASH_ARRAY(mime_types);
|
|
STORE_HASH_ARRAY(categories);
|
|
STORE_HASH_ARRAY(startup_wm_class);
|
|
STORE_HASH_ARRAY(name);
|
|
STORE_HASH_ARRAY(generic_name);
|
|
STORE_HASH_ARRAY(comment);
|
|
STORE_HASH_ARRAY(exec);
|
|
if (eina_hash_population(file_ids) > 0)
|
|
{
|
|
hash.hash = file_ids;
|
|
eet_data_write(util_ef, efreet_hash_string_edd(), "file_id", &hash, 1);
|
|
}
|
|
|
|
eina_hash_free(mime_types);
|
|
eina_hash_free(categories);
|
|
eina_hash_free(startup_wm_class);
|
|
eina_hash_free(name);
|
|
eina_hash_free(generic_name);
|
|
eina_hash_free(comment);
|
|
eina_hash_free(exec);
|
|
|
|
if (old_file_ids)
|
|
{
|
|
eina_hash_free(old_file_ids->hash);
|
|
free(old_file_ids);
|
|
}
|
|
|
|
eina_hash_free(file_ids);
|
|
eina_hash_free(paths);
|
|
|
|
eina_hash_free(desktops);
|
|
|
|
/* check if old and new caches contain the same number of entries */
|
|
if (!changed)
|
|
{
|
|
Eet_File *old;
|
|
|
|
old = eet_open(efreet_desktop_cache_file(), EET_FILE_MODE_READ);
|
|
if (!old || eet_num_entries(old) != eet_num_entries(ef)) changed = 1;
|
|
if (old) eet_close(old);
|
|
old = eet_open(efreet_desktop_util_cache_file(), EET_FILE_MODE_READ);
|
|
if (!old || eet_num_entries(old) != eet_num_entries(util_ef)) changed = 1;
|
|
if (old) eet_close(old);
|
|
}
|
|
|
|
/* cleanup */
|
|
eet_close(util_ef);
|
|
eet_close(ef);
|
|
|
|
/* unlink old cache files */
|
|
if (changed)
|
|
{
|
|
if (unlink(efreet_desktop_cache_file()) < 0)
|
|
{
|
|
if (errno != ENOENT) goto error;
|
|
}
|
|
if (unlink(efreet_desktop_util_cache_file()) < 0)
|
|
{
|
|
if (errno != ENOENT) goto error;
|
|
}
|
|
/* rename tmp files to real files */
|
|
if (rename(util_file, efreet_desktop_util_cache_file()) < 0) goto error;
|
|
efreet_setowner(efreet_desktop_util_cache_file());
|
|
if (rename(file, efreet_desktop_cache_file()) < 0) goto error;
|
|
efreet_setowner(efreet_desktop_cache_file());
|
|
}
|
|
else
|
|
{
|
|
unlink(util_file);
|
|
unlink(file);
|
|
}
|
|
|
|
{
|
|
char c = 'n';
|
|
|
|
if (changed) c = 'c';
|
|
printf("%c\n", c);
|
|
}
|
|
|
|
eina_tmpstr_del(file);
|
|
eina_tmpstr_del(util_file);
|
|
EINA_LIST_FREE(systemdirs, dir)
|
|
eina_stringshare_del(dir);
|
|
eina_list_free(extra_dirs);
|
|
efreet_shutdown();
|
|
ecore_shutdown();
|
|
eet_shutdown();
|
|
eina_log_domain_unregister(_efreet_desktop_cache_log_dom);
|
|
eina_shutdown();
|
|
close(lockfd);
|
|
return 0;
|
|
error:
|
|
eina_tmpstr_del(file);
|
|
eina_tmpstr_del(util_file);
|
|
IF_FREE(dir);
|
|
edd_error:
|
|
if (old_file_ids)
|
|
{
|
|
eina_hash_free(old_file_ids->hash);
|
|
free(old_file_ids);
|
|
}
|
|
efreet_shutdown();
|
|
efreet_error:
|
|
ecore_shutdown();
|
|
ecore_error:
|
|
eet_shutdown();
|
|
eet_error:
|
|
EINA_LIST_FREE(systemdirs, dir)
|
|
eina_stringshare_del(dir);
|
|
eina_list_free(extra_dirs);
|
|
eina_log_domain_unregister(_efreet_desktop_cache_log_dom);
|
|
eina_shutdown();
|
|
eina_error:
|
|
if (lockfd >= 0) close(lockfd);
|
|
return 1;
|
|
}
|