Import sugar-base into sugar-toolkit
Probably needs cleaning up a bit. And we use pygtk-codegen, ugh... This is the commit id when we imported sugar-base: b9406e5c9c9df5404c5b0d995178b5edb4d93628 Signed-off-by: Daniel Drake <dsd@laptop.org> [squashed two patches into one] Signed-off-by: Sascha Silbe <silbe@activitycentral.com>
This commit is contained in:
parent
b9a19e952f
commit
e2c07af748
@ -46,5 +46,6 @@ src/sugar3/bundle/Makefile
|
|||||||
src/sugar3/graphics/Makefile
|
src/sugar3/graphics/Makefile
|
||||||
src/sugar3/presence/Makefile
|
src/sugar3/presence/Makefile
|
||||||
src/sugar3/datastore/Makefile
|
src/sugar3/datastore/Makefile
|
||||||
|
src/sugar3/dispatch/Makefile
|
||||||
po/Makefile.in
|
po/Makefile.in
|
||||||
])
|
])
|
||||||
|
@ -1,8 +1,11 @@
|
|||||||
SUBDIRS = activity bundle graphics presence datastore
|
SUBDIRS = activity bundle graphics presence datastore dispatch
|
||||||
|
|
||||||
sugardir = $(pythondir)/sugar3
|
sugardir = $(pythondir)/sugar3
|
||||||
sugar_PYTHON = \
|
sugar_PYTHON = \
|
||||||
|
__init__.py \
|
||||||
env.py \
|
env.py \
|
||||||
|
logger.py \
|
||||||
|
mime.py \
|
||||||
network.py \
|
network.py \
|
||||||
profile.py \
|
profile.py \
|
||||||
session.py \
|
session.py \
|
||||||
@ -54,10 +57,49 @@ libsugarext_la_SOURCES = \
|
|||||||
sugar-menu.c \
|
sugar-menu.c \
|
||||||
sugar-menu.h
|
sugar-menu.h
|
||||||
|
|
||||||
|
sugar_LTLIBRARIES = _sugarbaseext.la
|
||||||
|
|
||||||
|
_sugarbaseext_la_CFLAGS = \
|
||||||
|
-DXDG_PREFIX=sugar_mime \
|
||||||
|
$(WARN_CFLAGS) \
|
||||||
|
$(EXT_CFLAGS) \
|
||||||
|
$(PYTHON_INCLUDES)
|
||||||
|
|
||||||
BUILT_SOURCES = \
|
BUILT_SOURCES = \
|
||||||
sugar-marshal.c \
|
sugar-marshal.c \
|
||||||
sugar-marshal.h
|
sugar-marshal.h
|
||||||
|
|
||||||
|
_sugarbaseext_la_LDFLAGS = -module -avoid-version
|
||||||
|
_sugarbaseext_la_LIBADD = $(EXT_LIBS)
|
||||||
|
_sugarbaseext_la_SOURCES = \
|
||||||
|
_sugarbaseextmodule.c \
|
||||||
|
xdgmime.c \
|
||||||
|
xdgmime.h \
|
||||||
|
xdgmimealias.c \
|
||||||
|
xdgmimealias.h \
|
||||||
|
xdgmimecache.c \
|
||||||
|
xdgmimecache.h \
|
||||||
|
xdgmimeglob.c \
|
||||||
|
xdgmimeglob.h \
|
||||||
|
xdgmimeint.c \
|
||||||
|
xdgmimeint.h \
|
||||||
|
xdgmimemagic.c \
|
||||||
|
xdgmimemagic.h \
|
||||||
|
xdgmimeparent.c \
|
||||||
|
xdgmimeparent.h
|
||||||
|
|
||||||
|
nodist__sugarbaseext_la_SOURCES = _sugarbaseext.c
|
||||||
|
|
||||||
|
_sugarbaseext.c: _sugarbaseext.defs _sugarbaseext.override
|
||||||
|
|
||||||
|
.defs.c:
|
||||||
|
(cd $(srcdir)\
|
||||||
|
&& $(PYGTK_CODEGEN) \
|
||||||
|
--override $*.override \
|
||||||
|
--prefix py$* $*.defs) > gen-$*.c \
|
||||||
|
&& cp gen-$*.c $*.c \
|
||||||
|
&& rm -f gen-$*.c
|
||||||
|
|
||||||
sugar-marshal.c: sugar-marshal.list
|
sugar-marshal.c: sugar-marshal.list
|
||||||
$(GLIB_GENMARSHAL) --prefix=sugar_marshal \
|
$(GLIB_GENMARSHAL) --prefix=sugar_marshal \
|
||||||
$(srcdir)/sugar-marshal.list --header --body > sugar-marshal.c
|
$(srcdir)/sugar-marshal.list --header --body > sugar-marshal.c
|
||||||
@ -66,8 +108,8 @@ sugar-marshal.h: sugar-marshal.list
|
|||||||
$(GLIB_GENMARSHAL) --prefix=sugar_marshal \
|
$(GLIB_GENMARSHAL) --prefix=sugar_marshal \
|
||||||
$(srcdir)/sugar-marshal.list --header > sugar-marshal.h
|
$(srcdir)/sugar-marshal.list --header > sugar-marshal.h
|
||||||
|
|
||||||
CLEANFILES = $(BUILT_SOURCES)
|
CLEANFILES = $(BUILT_SOURCES) _sugarbaseext.c
|
||||||
EXTRA_DIST = sugar-marshal.list
|
EXTRA_DIST = sugar-marshal.list _sugarbaseext.override _sugarbaseext.defs
|
||||||
|
|
||||||
-include $(INTROSPECTION_MAKEFILE)
|
-include $(INTROSPECTION_MAKEFILE)
|
||||||
INTROSPECTION_GIRS = SugarExt-1.0.gir
|
INTROSPECTION_GIRS = SugarExt-1.0.gir
|
||||||
|
30
src/sugar3/__init__.py
Normal file
30
src/sugar3/__init__.py
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
# Copyright (C) 2006-2007, Red Hat, Inc.
|
||||||
|
# Copyright (C) 2007-2008, One Laptop Per Child
|
||||||
|
#
|
||||||
|
# This library 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; either
|
||||||
|
# version 2 of the License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This library 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
|
||||||
|
# Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public
|
||||||
|
# License along with this library; if not, write to the
|
||||||
|
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
# Boston, MA 02111-1307, USA.
|
||||||
|
|
||||||
|
import os
|
||||||
|
import gettext
|
||||||
|
|
||||||
|
|
||||||
|
if 'SUGAR_PREFIX' in os.environ:
|
||||||
|
prefix = os.environ['SUGAR_PREFIX']
|
||||||
|
else:
|
||||||
|
prefix = '/usr'
|
||||||
|
|
||||||
|
locale_path = os.path.join(prefix, 'share', 'locale')
|
||||||
|
|
||||||
|
gettext.bindtextdomain('sugar-base', locale_path)
|
33
src/sugar3/_sugarbaseext.defs
Normal file
33
src/sugar3/_sugarbaseext.defs
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
; functions
|
||||||
|
|
||||||
|
(define-function get_mime_type_from_file_name
|
||||||
|
(c-name "sugar_mime_get_mime_type_from_file_name")
|
||||||
|
(return-type "const-char*")
|
||||||
|
(parameters
|
||||||
|
'("const-char*" "filename")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
(define-function get_mime_type_for_file
|
||||||
|
(c-name "sugar_mime_get_mime_type_for_file")
|
||||||
|
(return-type "const-char*")
|
||||||
|
(parameters
|
||||||
|
'("const-char*" "filename")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
(define-function list_mime_parents
|
||||||
|
(c-name "sugar_mime_list_mime_parents")
|
||||||
|
(return-type "char**")
|
||||||
|
(parameters
|
||||||
|
'("const-char*" "mime")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
(define-function uri_list_extract_uris
|
||||||
|
(c-name "g_uri_list_extract_uris")
|
||||||
|
(return-type "gchar**")
|
||||||
|
(parameters
|
||||||
|
'("const-char*" "uri_list")
|
||||||
|
)
|
||||||
|
)
|
88
src/sugar3/_sugarbaseext.override
Normal file
88
src/sugar3/_sugarbaseext.override
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
/* -*- Mode: C; c-basic-offset: 4 -*- */
|
||||||
|
%%
|
||||||
|
headers
|
||||||
|
#include <Python.h>
|
||||||
|
#include <glib.h>
|
||||||
|
#include "xdgmime.h"
|
||||||
|
%%
|
||||||
|
modulename _sugarext
|
||||||
|
%%
|
||||||
|
ignore-glob
|
||||||
|
*_get_type
|
||||||
|
_*
|
||||||
|
%%
|
||||||
|
override sugar_mime_get_mime_type_for_file kwargs
|
||||||
|
static PyObject *
|
||||||
|
_wrap_sugar_mime_get_mime_type_for_file(PyObject *self, PyObject *args, PyObject *kwargs)
|
||||||
|
{
|
||||||
|
static char *kwlist[] = { "filename", NULL };
|
||||||
|
char *filename;
|
||||||
|
const char *ret;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTupleAndKeywords(args, kwargs,"s:get_mime_type_for_file", kwlist, &filename))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
ret = sugar_mime_get_mime_type_for_file(filename, NULL);
|
||||||
|
|
||||||
|
if (ret)
|
||||||
|
return PyString_FromString(ret);
|
||||||
|
Py_INCREF(Py_None);
|
||||||
|
return Py_None;
|
||||||
|
}
|
||||||
|
%%
|
||||||
|
override sugar_mime_list_mime_parents kwargs
|
||||||
|
static PyObject *
|
||||||
|
_wrap_sugar_mime_list_mime_parents(PyObject *self, PyObject *args, PyObject *kwargs)
|
||||||
|
{
|
||||||
|
static char *kwlist[] = { "mime_type", NULL };
|
||||||
|
char *mime_type;
|
||||||
|
char **parents, **tmp;
|
||||||
|
int i = 0, j;
|
||||||
|
PyObject *ret;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTupleAndKeywords(args, kwargs,"s:list_mime_parents", kwlist, &mime_type))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
parents = (char **)sugar_mime_list_mime_parents(mime_type);
|
||||||
|
if (!parents)
|
||||||
|
return PyTuple_New(0);
|
||||||
|
|
||||||
|
tmp = parents;
|
||||||
|
while (*tmp)
|
||||||
|
tmp++, i++;
|
||||||
|
|
||||||
|
ret = PyTuple_New(i);
|
||||||
|
for (j = 0; j < i; j++)
|
||||||
|
PyTuple_SetItem(ret, j, PyString_FromString(parents[j]));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
%%
|
||||||
|
override g_uri_list_extract_uris kwargs
|
||||||
|
static PyObject *
|
||||||
|
_wrap_g_uri_list_extract_uris(PyObject *self, PyObject *args, PyObject *kwargs)
|
||||||
|
{
|
||||||
|
static char *kwlist[] = { "uri_list", NULL };
|
||||||
|
char *uri_list;
|
||||||
|
char **uris, **tmp;
|
||||||
|
int i = 0, j;
|
||||||
|
PyObject *ret;
|
||||||
|
|
||||||
|
if (!PyArg_ParseTupleAndKeywords(args, kwargs,"s:uri_list_extract_uris", kwlist, &uri_list))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
uris = (char **)g_uri_list_extract_uris(uri_list);
|
||||||
|
if (!uris)
|
||||||
|
return PyTuple_New(0);
|
||||||
|
|
||||||
|
tmp = uris;
|
||||||
|
while (*tmp)
|
||||||
|
tmp++, i++;
|
||||||
|
|
||||||
|
ret = PyTuple_New(i);
|
||||||
|
for (j = 0; j < i; j++)
|
||||||
|
PyTuple_SetItem(ret, j, PyString_FromString(uris[j]));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
%%
|
42
src/sugar3/_sugarbaseextmodule.c
Normal file
42
src/sugar3/_sugarbaseextmodule.c
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2006-2007, Red Hat, Inc.
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include "config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* include this first, before NO_IMPORT_PYGOBJECT is defined */
|
||||||
|
#include <pygobject.h>
|
||||||
|
|
||||||
|
extern PyMethodDef py_sugarbaseext_functions[];
|
||||||
|
|
||||||
|
DL_EXPORT(void)
|
||||||
|
init_sugarbaseext(void)
|
||||||
|
{
|
||||||
|
PyObject *m, *d;
|
||||||
|
|
||||||
|
init_pygobject ();
|
||||||
|
|
||||||
|
m = Py_InitModule ("_sugarbaseext", py_sugarbaseext_functions);
|
||||||
|
d = PyModule_GetDict (m);
|
||||||
|
|
||||||
|
if (PyErr_Occurred ()) {
|
||||||
|
Py_FatalError ("can't initialise module _sugarext");
|
||||||
|
}
|
||||||
|
}
|
9
src/sugar3/dispatch/Makefile.am
Normal file
9
src/sugar3/dispatch/Makefile.am
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
sugardir = $(pythondir)/sugar3/dispatch
|
||||||
|
sugar_PYTHON = \
|
||||||
|
__init__.py \
|
||||||
|
dispatcher.py \
|
||||||
|
saferef.py
|
||||||
|
|
||||||
|
EXTRA_DIST = \
|
||||||
|
license.txt
|
||||||
|
|
9
src/sugar3/dispatch/__init__.py
Normal file
9
src/sugar3/dispatch/__init__.py
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
"""Multi-consumer multi-producer dispatching mechanism
|
||||||
|
|
||||||
|
Originally based on pydispatch (BSD) http://pypi.python.org/pypi/PyDispatcher/2.0.1
|
||||||
|
See license.txt for original license.
|
||||||
|
|
||||||
|
Heavily modified for Django's purposes.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from sugar.dispatch.dispatcher import Signal
|
191
src/sugar3/dispatch/dispatcher.py
Normal file
191
src/sugar3/dispatch/dispatcher.py
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
import weakref
|
||||||
|
try:
|
||||||
|
set
|
||||||
|
except NameError:
|
||||||
|
from sets import Set as set # Python 2.3 fallback
|
||||||
|
|
||||||
|
from sugar.dispatch import saferef
|
||||||
|
|
||||||
|
WEAKREF_TYPES = (weakref.ReferenceType, saferef.BoundMethodWeakref)
|
||||||
|
|
||||||
|
def _make_id(target):
|
||||||
|
if hasattr(target, 'im_func'):
|
||||||
|
return (id(target.im_self), id(target.im_func))
|
||||||
|
return id(target)
|
||||||
|
|
||||||
|
class Signal(object):
|
||||||
|
"""Base class for all signals
|
||||||
|
|
||||||
|
Internal attributes:
|
||||||
|
receivers -- { receriverkey (id) : weakref(receiver) }
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, providing_args=None):
|
||||||
|
"""providing_args -- A list of the arguments this signal can pass along in
|
||||||
|
a send() call.
|
||||||
|
"""
|
||||||
|
self.receivers = []
|
||||||
|
if providing_args is None:
|
||||||
|
providing_args = []
|
||||||
|
self.providing_args = set(providing_args)
|
||||||
|
|
||||||
|
def connect(self, receiver, sender=None, weak=True, dispatch_uid=None):
|
||||||
|
"""Connect receiver to sender for signal
|
||||||
|
|
||||||
|
receiver -- a function or an instance method which is to
|
||||||
|
receive signals. Receivers must be
|
||||||
|
hashable objects.
|
||||||
|
|
||||||
|
if weak is True, then receiver must be weak-referencable
|
||||||
|
(more precisely saferef.safeRef() must be able to create
|
||||||
|
a reference to the receiver).
|
||||||
|
|
||||||
|
Receivers must be able to accept keyword arguments.
|
||||||
|
|
||||||
|
If receivers have a dispatch_uid attribute, the receiver will
|
||||||
|
not be added if another receiver already exists with that
|
||||||
|
dispatch_uid.
|
||||||
|
|
||||||
|
sender -- the sender to which the receiver should respond
|
||||||
|
Must either be of type Signal, or None to receive events
|
||||||
|
from any sender.
|
||||||
|
|
||||||
|
weak -- whether to use weak references to the receiver
|
||||||
|
By default, the module will attempt to use weak
|
||||||
|
references to the receiver objects. If this parameter
|
||||||
|
is false, then strong references will be used.
|
||||||
|
|
||||||
|
dispatch_uid -- an identifier used to uniquely identify a particular
|
||||||
|
instance of a receiver. This will usually be a string, though it
|
||||||
|
may be anything hashable.
|
||||||
|
|
||||||
|
returns None
|
||||||
|
"""
|
||||||
|
if dispatch_uid:
|
||||||
|
lookup_key = (dispatch_uid, _make_id(sender))
|
||||||
|
else:
|
||||||
|
lookup_key = (_make_id(receiver), _make_id(sender))
|
||||||
|
|
||||||
|
if weak:
|
||||||
|
receiver = saferef.safeRef(receiver, onDelete=self._remove_receiver)
|
||||||
|
|
||||||
|
for r_key, _ in self.receivers:
|
||||||
|
if r_key == lookup_key:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
self.receivers.append((lookup_key, receiver))
|
||||||
|
|
||||||
|
def disconnect(self, receiver=None, sender=None, weak=True, dispatch_uid=None):
|
||||||
|
"""Disconnect receiver from sender for signal
|
||||||
|
|
||||||
|
receiver -- the registered receiver to disconnect. May be none if
|
||||||
|
dispatch_uid is specified.
|
||||||
|
sender -- the registered sender to disconnect
|
||||||
|
weak -- the weakref state to disconnect
|
||||||
|
dispatch_uid -- the unique identifier of the receiver to disconnect
|
||||||
|
|
||||||
|
disconnect reverses the process of connect.
|
||||||
|
|
||||||
|
If weak references are used, disconnect need not be called.
|
||||||
|
The receiver will be remove from dispatch automatically.
|
||||||
|
|
||||||
|
returns None
|
||||||
|
"""
|
||||||
|
|
||||||
|
if dispatch_uid:
|
||||||
|
lookup_key = (dispatch_uid, _make_id(sender))
|
||||||
|
else:
|
||||||
|
lookup_key = (_make_id(receiver), _make_id(sender))
|
||||||
|
|
||||||
|
for idx, (r_key, _) in enumerate(self.receivers):
|
||||||
|
if r_key == lookup_key:
|
||||||
|
del self.receivers[idx]
|
||||||
|
|
||||||
|
def send(self, sender, **named):
|
||||||
|
"""Send signal from sender to all connected receivers.
|
||||||
|
|
||||||
|
sender -- the sender of the signal
|
||||||
|
Either a specific object or None.
|
||||||
|
|
||||||
|
named -- named arguments which will be passed to receivers.
|
||||||
|
|
||||||
|
Returns a list of tuple pairs [(receiver, response), ... ].
|
||||||
|
|
||||||
|
If any receiver raises an error, the error propagates back
|
||||||
|
through send, terminating the dispatch loop, so it is quite
|
||||||
|
possible to not have all receivers called if a raises an
|
||||||
|
error.
|
||||||
|
"""
|
||||||
|
|
||||||
|
responses = []
|
||||||
|
if not self.receivers:
|
||||||
|
return responses
|
||||||
|
|
||||||
|
for receiver in self._live_receivers(_make_id(sender)):
|
||||||
|
response = receiver(signal=self, sender=sender, **named)
|
||||||
|
responses.append((receiver, response))
|
||||||
|
return responses
|
||||||
|
|
||||||
|
def send_robust(self, sender, **named):
|
||||||
|
"""Send signal from sender to all connected receivers catching errors
|
||||||
|
|
||||||
|
sender -- the sender of the signal
|
||||||
|
Can be any python object (normally one registered with
|
||||||
|
a connect if you actually want something to occur).
|
||||||
|
|
||||||
|
named -- named arguments which will be passed to receivers.
|
||||||
|
These arguments must be a subset of the argument names
|
||||||
|
defined in providing_args.
|
||||||
|
|
||||||
|
Return a list of tuple pairs [(receiver, response), ... ],
|
||||||
|
may raise DispatcherKeyError
|
||||||
|
|
||||||
|
if any receiver raises an error (specifically any subclass of Exception),
|
||||||
|
the error instance is returned as the result for that receiver.
|
||||||
|
"""
|
||||||
|
|
||||||
|
responses = []
|
||||||
|
if not self.receivers:
|
||||||
|
return responses
|
||||||
|
|
||||||
|
# Call each receiver with whatever arguments it can accept.
|
||||||
|
# Return a list of tuple pairs [(receiver, response), ... ].
|
||||||
|
for receiver in self._live_receivers(_make_id(sender)):
|
||||||
|
try:
|
||||||
|
response = receiver(signal=self, sender=sender, **named)
|
||||||
|
except Exception, err:
|
||||||
|
responses.append((receiver, err))
|
||||||
|
else:
|
||||||
|
responses.append((receiver, response))
|
||||||
|
return responses
|
||||||
|
|
||||||
|
def _live_receivers(self, senderkey):
|
||||||
|
"""Filter sequence of receivers to get resolved, live receivers
|
||||||
|
|
||||||
|
This checks for weak references
|
||||||
|
and resolves them, then returning only live
|
||||||
|
receivers.
|
||||||
|
"""
|
||||||
|
none_senderkey = _make_id(None)
|
||||||
|
|
||||||
|
for (receiverkey, r_senderkey), receiver in self.receivers:
|
||||||
|
if r_senderkey == none_senderkey or r_senderkey == senderkey:
|
||||||
|
if isinstance(receiver, WEAKREF_TYPES):
|
||||||
|
# Dereference the weak reference.
|
||||||
|
receiver = receiver()
|
||||||
|
if receiver is not None:
|
||||||
|
yield receiver
|
||||||
|
else:
|
||||||
|
yield receiver
|
||||||
|
|
||||||
|
def _remove_receiver(self, receiver):
|
||||||
|
"""Remove dead receivers from connections."""
|
||||||
|
|
||||||
|
to_remove = []
|
||||||
|
for key, connected_receiver in self.receivers:
|
||||||
|
if connected_receiver == receiver:
|
||||||
|
to_remove.append(key)
|
||||||
|
for key in to_remove:
|
||||||
|
for idx, (r_key, _) in enumerate(self.receivers):
|
||||||
|
if r_key == key:
|
||||||
|
del self.receivers[idx]
|
66
src/sugar3/dispatch/license.txt
Normal file
66
src/sugar3/dispatch/license.txt
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
sugar.dispatch was originally forked from django.dispatch
|
||||||
|
|
||||||
|
Copyright (c) Django Software Foundation and individual contributors.
|
||||||
|
All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
this list of conditions and the following disclaimer.
|
||||||
|
|
||||||
|
2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
notice, this list of conditions and the following disclaimer in the
|
||||||
|
documentation and/or other materials provided with the distribution.
|
||||||
|
|
||||||
|
3. Neither the name of Django nor the names of its contributors may be used
|
||||||
|
to endorse or promote products derived from this software without
|
||||||
|
specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
||||||
|
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
||||||
|
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
django.dispatch was originally forked from PyDispatcher.
|
||||||
|
|
||||||
|
PyDispatcher License:
|
||||||
|
|
||||||
|
Copyright (c) 2001-2003, Patrick K. O'Brien and Contributors
|
||||||
|
All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
modification, are permitted provided that the following conditions
|
||||||
|
are met:
|
||||||
|
|
||||||
|
Redistributions of source code must retain the above copyright
|
||||||
|
notice, this list of conditions and the following disclaimer.
|
||||||
|
|
||||||
|
Redistributions in binary form must reproduce the above
|
||||||
|
copyright notice, this list of conditions and the following
|
||||||
|
disclaimer in the documentation and/or other materials
|
||||||
|
provided with the distribution.
|
||||||
|
|
||||||
|
The name of Patrick K. O'Brien, or the name of any Contributor,
|
||||||
|
may not be used to endorse or promote products derived from this
|
||||||
|
software without specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||||
|
COPYRIGHT HOLDERS AND CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
||||||
|
INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||||||
|
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||||
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
||||||
|
OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
250
src/sugar3/dispatch/saferef.py
Normal file
250
src/sugar3/dispatch/saferef.py
Normal file
@ -0,0 +1,250 @@
|
|||||||
|
"""
|
||||||
|
"Safe weakrefs", originally from pyDispatcher.
|
||||||
|
|
||||||
|
Provides a way to safely weakref any function, including bound methods (which
|
||||||
|
aren't handled by the core weakref module).
|
||||||
|
"""
|
||||||
|
|
||||||
|
import weakref, traceback
|
||||||
|
|
||||||
|
def safeRef(target, onDelete = None):
|
||||||
|
"""Return a *safe* weak reference to a callable target
|
||||||
|
|
||||||
|
target -- the object to be weakly referenced, if it's a
|
||||||
|
bound method reference, will create a BoundMethodWeakref,
|
||||||
|
otherwise creates a simple weakref.
|
||||||
|
onDelete -- if provided, will have a hard reference stored
|
||||||
|
to the callable to be called after the safe reference
|
||||||
|
goes out of scope with the reference object, (either a
|
||||||
|
weakref or a BoundMethodWeakref) as argument.
|
||||||
|
"""
|
||||||
|
if hasattr(target, 'im_self'):
|
||||||
|
if target.im_self is not None:
|
||||||
|
# Turn a bound method into a BoundMethodWeakref instance.
|
||||||
|
# Keep track of these instances for lookup by disconnect().
|
||||||
|
assert hasattr(target, 'im_func'), """safeRef target %r has im_self, but no im_func, don't know how to create reference"""%( target,)
|
||||||
|
reference = get_bound_method_weakref(
|
||||||
|
target=target,
|
||||||
|
onDelete=onDelete
|
||||||
|
)
|
||||||
|
return reference
|
||||||
|
if callable(onDelete):
|
||||||
|
return weakref.ref(target, onDelete)
|
||||||
|
else:
|
||||||
|
return weakref.ref( target )
|
||||||
|
|
||||||
|
class BoundMethodWeakref(object):
|
||||||
|
"""'Safe' and reusable weak references to instance methods
|
||||||
|
|
||||||
|
BoundMethodWeakref objects provide a mechanism for
|
||||||
|
referencing a bound method without requiring that the
|
||||||
|
method object itself (which is normally a transient
|
||||||
|
object) is kept alive. Instead, the BoundMethodWeakref
|
||||||
|
object keeps weak references to both the object and the
|
||||||
|
function which together define the instance method.
|
||||||
|
|
||||||
|
Attributes:
|
||||||
|
key -- the identity key for the reference, calculated
|
||||||
|
by the class's calculateKey method applied to the
|
||||||
|
target instance method
|
||||||
|
deletionMethods -- sequence of callable objects taking
|
||||||
|
single argument, a reference to this object which
|
||||||
|
will be called when *either* the target object or
|
||||||
|
target function is garbage collected (i.e. when
|
||||||
|
this object becomes invalid). These are specified
|
||||||
|
as the onDelete parameters of safeRef calls.
|
||||||
|
weakSelf -- weak reference to the target object
|
||||||
|
weakFunc -- weak reference to the target function
|
||||||
|
|
||||||
|
Class Attributes:
|
||||||
|
_allInstances -- class attribute pointing to all live
|
||||||
|
BoundMethodWeakref objects indexed by the class's
|
||||||
|
calculateKey(target) method applied to the target
|
||||||
|
objects. This weak value dictionary is used to
|
||||||
|
short-circuit creation so that multiple references
|
||||||
|
to the same (object, function) pair produce the
|
||||||
|
same BoundMethodWeakref instance.
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
_allInstances = weakref.WeakValueDictionary()
|
||||||
|
|
||||||
|
def __new__( cls, target, onDelete=None, *arguments,**named ):
|
||||||
|
"""Create new instance or return current instance
|
||||||
|
|
||||||
|
Basically this method of construction allows us to
|
||||||
|
short-circuit creation of references to already-
|
||||||
|
referenced instance methods. The key corresponding
|
||||||
|
to the target is calculated, and if there is already
|
||||||
|
an existing reference, that is returned, with its
|
||||||
|
deletionMethods attribute updated. Otherwise the
|
||||||
|
new instance is created and registered in the table
|
||||||
|
of already-referenced methods.
|
||||||
|
"""
|
||||||
|
key = cls.calculateKey(target)
|
||||||
|
current =cls._allInstances.get(key)
|
||||||
|
if current is not None:
|
||||||
|
current.deletionMethods.append( onDelete)
|
||||||
|
return current
|
||||||
|
else:
|
||||||
|
base = super( BoundMethodWeakref, cls).__new__( cls )
|
||||||
|
cls._allInstances[key] = base
|
||||||
|
base.__init__( target, onDelete, *arguments,**named)
|
||||||
|
return base
|
||||||
|
|
||||||
|
def __init__(self, target, onDelete=None):
|
||||||
|
"""Return a weak-reference-like instance for a bound method
|
||||||
|
|
||||||
|
target -- the instance-method target for the weak
|
||||||
|
reference, must have im_self and im_func attributes
|
||||||
|
and be reconstructable via:
|
||||||
|
target.im_func.__get__( target.im_self )
|
||||||
|
which is true of built-in instance methods.
|
||||||
|
onDelete -- optional callback which will be called
|
||||||
|
when this weak reference ceases to be valid
|
||||||
|
(i.e. either the object or the function is garbage
|
||||||
|
collected). Should take a single argument,
|
||||||
|
which will be passed a pointer to this object.
|
||||||
|
"""
|
||||||
|
def remove(weak, self=self):
|
||||||
|
"""Set self.isDead to true when method or instance is destroyed"""
|
||||||
|
methods = self.deletionMethods[:]
|
||||||
|
del self.deletionMethods[:]
|
||||||
|
try:
|
||||||
|
del self.__class__._allInstances[ self.key ]
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
for function in methods:
|
||||||
|
try:
|
||||||
|
if callable( function ):
|
||||||
|
function( self )
|
||||||
|
except Exception, e:
|
||||||
|
try:
|
||||||
|
traceback.print_exc()
|
||||||
|
except AttributeError, err:
|
||||||
|
print '''Exception during saferef %s cleanup function %s: %s'''%(
|
||||||
|
self, function, e
|
||||||
|
)
|
||||||
|
self.deletionMethods = [onDelete]
|
||||||
|
self.key = self.calculateKey( target )
|
||||||
|
self.weakSelf = weakref.ref(target.im_self, remove)
|
||||||
|
self.weakFunc = weakref.ref(target.im_func, remove)
|
||||||
|
self.selfName = str(target.im_self)
|
||||||
|
self.funcName = str(target.im_func.__name__)
|
||||||
|
|
||||||
|
def calculateKey( cls, target ):
|
||||||
|
"""Calculate the reference key for this reference
|
||||||
|
|
||||||
|
Currently this is a two-tuple of the id()'s of the
|
||||||
|
target object and the target function respectively.
|
||||||
|
"""
|
||||||
|
return (id(target.im_self),id(target.im_func))
|
||||||
|
calculateKey = classmethod( calculateKey )
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
"""Give a friendly representation of the object"""
|
||||||
|
return """%s( %s.%s )"""%(
|
||||||
|
self.__class__.__name__,
|
||||||
|
self.selfName,
|
||||||
|
self.funcName,
|
||||||
|
)
|
||||||
|
|
||||||
|
__repr__ = __str__
|
||||||
|
|
||||||
|
def __nonzero__( self ):
|
||||||
|
"""Whether we are still a valid reference"""
|
||||||
|
return self() is not None
|
||||||
|
|
||||||
|
def __cmp__( self, other ):
|
||||||
|
"""Compare with another reference"""
|
||||||
|
if not isinstance (other,self.__class__):
|
||||||
|
return cmp( self.__class__, type(other) )
|
||||||
|
return cmp( self.key, other.key)
|
||||||
|
|
||||||
|
def __call__(self):
|
||||||
|
"""Return a strong reference to the bound method
|
||||||
|
|
||||||
|
If the target cannot be retrieved, then will
|
||||||
|
return None, otherwise returns a bound instance
|
||||||
|
method for our object and function.
|
||||||
|
|
||||||
|
Note:
|
||||||
|
You may call this method any number of times,
|
||||||
|
as it does not invalidate the reference.
|
||||||
|
"""
|
||||||
|
target = self.weakSelf()
|
||||||
|
if target is not None:
|
||||||
|
function = self.weakFunc()
|
||||||
|
if function is not None:
|
||||||
|
return function.__get__(target)
|
||||||
|
return None
|
||||||
|
|
||||||
|
class BoundNonDescriptorMethodWeakref(BoundMethodWeakref):
|
||||||
|
"""A specialized BoundMethodWeakref, for platforms where instance methods
|
||||||
|
are not descriptors.
|
||||||
|
|
||||||
|
It assumes that the function name and the target attribute name are the
|
||||||
|
same, instead of assuming that the function is a descriptor. This approach
|
||||||
|
is equally fast, but not 100% reliable because functions can be stored on an
|
||||||
|
attribute named differenty than the function's name such as in:
|
||||||
|
|
||||||
|
class A: pass
|
||||||
|
def foo(self): return "foo"
|
||||||
|
A.bar = foo
|
||||||
|
|
||||||
|
But this shouldn't be a common use case. So, on platforms where methods
|
||||||
|
aren't descriptors (such as Jython) this implementation has the advantage
|
||||||
|
of working in the most cases.
|
||||||
|
"""
|
||||||
|
def __init__(self, target, onDelete=None):
|
||||||
|
"""Return a weak-reference-like instance for a bound method
|
||||||
|
|
||||||
|
target -- the instance-method target for the weak
|
||||||
|
reference, must have im_self and im_func attributes
|
||||||
|
and be reconstructable via:
|
||||||
|
target.im_func.__get__( target.im_self )
|
||||||
|
which is true of built-in instance methods.
|
||||||
|
onDelete -- optional callback which will be called
|
||||||
|
when this weak reference ceases to be valid
|
||||||
|
(i.e. either the object or the function is garbage
|
||||||
|
collected). Should take a single argument,
|
||||||
|
which will be passed a pointer to this object.
|
||||||
|
"""
|
||||||
|
assert getattr(target.im_self, target.__name__) == target, \
|
||||||
|
("method %s isn't available as the attribute %s of %s" %
|
||||||
|
(target, target.__name__, target.im_self))
|
||||||
|
super(BoundNonDescriptorMethodWeakref, self).__init__(target, onDelete)
|
||||||
|
|
||||||
|
def __call__(self):
|
||||||
|
"""Return a strong reference to the bound method
|
||||||
|
|
||||||
|
If the target cannot be retrieved, then will
|
||||||
|
return None, otherwise returns a bound instance
|
||||||
|
method for our object and function.
|
||||||
|
|
||||||
|
Note:
|
||||||
|
You may call this method any number of times,
|
||||||
|
as it does not invalidate the reference.
|
||||||
|
"""
|
||||||
|
target = self.weakSelf()
|
||||||
|
if target is not None:
|
||||||
|
function = self.weakFunc()
|
||||||
|
if function is not None:
|
||||||
|
# Using curry() would be another option, but it erases the
|
||||||
|
# "signature" of the function. That is, after a function is
|
||||||
|
# curried, the inspect module can't be used to determine how
|
||||||
|
# many arguments the function expects, nor what keyword
|
||||||
|
# arguments it supports, and pydispatcher needs this
|
||||||
|
# information.
|
||||||
|
return getattr(target, function.__name__)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def get_bound_method_weakref(target, onDelete):
|
||||||
|
"""Instantiates the appropiate BoundMethodWeakRef, depending on the details of
|
||||||
|
the underlying class method implementation"""
|
||||||
|
if hasattr(target, '__get__'):
|
||||||
|
# target method is a descriptor, so the default implementation works:
|
||||||
|
return BoundMethodWeakref(target=target, onDelete=onDelete)
|
||||||
|
else:
|
||||||
|
# no luck, use the alternative implementation:
|
||||||
|
return BoundNonDescriptorMethodWeakref(target=target, onDelete=onDelete)
|
199
src/sugar3/logger.py
Normal file
199
src/sugar3/logger.py
Normal file
@ -0,0 +1,199 @@
|
|||||||
|
# Copyright (C) 2007 Red Hat, Inc.
|
||||||
|
#
|
||||||
|
# This library 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; either
|
||||||
|
# version 2 of the License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This library 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
|
||||||
|
# Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public
|
||||||
|
# License along with this library; if not, write to the
|
||||||
|
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
# Boston, MA 02111-1307, USA.
|
||||||
|
|
||||||
|
"""Logging service setup.
|
||||||
|
|
||||||
|
STABLE.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import array
|
||||||
|
import collections
|
||||||
|
import errno
|
||||||
|
import logging
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
import repr as repr_
|
||||||
|
import decorator
|
||||||
|
|
||||||
|
# Let's keep this module self contained so that it can be easily
|
||||||
|
# pasted in external sugar service like the datastore.
|
||||||
|
|
||||||
|
# traces function calls, use SUGAR_LOGGER_LEVEL=trace to enable
|
||||||
|
TRACE = 5
|
||||||
|
_LEVELS = {
|
||||||
|
'error': logging.ERROR,
|
||||||
|
'warning': logging.WARNING,
|
||||||
|
'debug': logging.DEBUG,
|
||||||
|
'info': logging.INFO,
|
||||||
|
'trace': TRACE,
|
||||||
|
'all': 0,
|
||||||
|
}
|
||||||
|
logging.addLevelName(TRACE, 'TRACE')
|
||||||
|
|
||||||
|
|
||||||
|
def get_logs_dir():
|
||||||
|
profile = os.environ.get('SUGAR_PROFILE', 'default')
|
||||||
|
logs_dir = os.environ.get('SUGAR_LOGS_DIR',
|
||||||
|
os.path.join(os.path.expanduser('~'),
|
||||||
|
'.sugar', profile, 'logs'))
|
||||||
|
return logs_dir
|
||||||
|
|
||||||
|
|
||||||
|
def set_level(level):
|
||||||
|
if level in _LEVELS:
|
||||||
|
logging.getLogger('').setLevel(_LEVELS[level])
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
logging.getLogger('').setLevel(int(level))
|
||||||
|
except ValueError:
|
||||||
|
logging.warning('Invalid log level: %r', level)
|
||||||
|
|
||||||
|
|
||||||
|
# pylint: disable-msg=E1101,F0401
|
||||||
|
def _except_hook(exctype, value, traceback):
|
||||||
|
# Attempt to provide verbose IPython tracebacks.
|
||||||
|
# Importing IPython is slow, so we import it lazily.
|
||||||
|
try:
|
||||||
|
from IPython.ultraTB import AutoFormattedTB
|
||||||
|
sys.excepthook = AutoFormattedTB(mode='Verbose',
|
||||||
|
color_scheme='NoColor')
|
||||||
|
except ImportError:
|
||||||
|
sys.excepthook = sys.__excepthook__
|
||||||
|
|
||||||
|
sys.excepthook(exctype, value, traceback)
|
||||||
|
|
||||||
|
|
||||||
|
def start(log_filename=None):
|
||||||
|
# remove existing handlers, or logging.basicConfig() won't have no effect.
|
||||||
|
root_logger = logging.getLogger('')
|
||||||
|
for handler in root_logger.handlers:
|
||||||
|
root_logger.removeHandler(handler)
|
||||||
|
|
||||||
|
class SafeLogWrapper(object):
|
||||||
|
"""Small file-like wrapper to gracefully handle ENOSPC errors when
|
||||||
|
logging."""
|
||||||
|
|
||||||
|
def __init__(self, stream):
|
||||||
|
self._stream = stream
|
||||||
|
|
||||||
|
def write(self, s):
|
||||||
|
try:
|
||||||
|
self._stream.write(s)
|
||||||
|
except IOError, e:
|
||||||
|
# gracefully deal w/ disk full
|
||||||
|
if e.errno != errno.ENOSPC:
|
||||||
|
raise e
|
||||||
|
|
||||||
|
def flush(self):
|
||||||
|
try:
|
||||||
|
self._stream.flush()
|
||||||
|
except IOError, e:
|
||||||
|
# gracefully deal w/ disk full
|
||||||
|
if e.errno != errno.ENOSPC:
|
||||||
|
raise e
|
||||||
|
|
||||||
|
logging.basicConfig(level=logging.WARNING,
|
||||||
|
format="%(created)f %(levelname)s %(name)s: %(message)s",
|
||||||
|
stream=SafeLogWrapper(sys.stderr))
|
||||||
|
|
||||||
|
if 'SUGAR_LOGGER_LEVEL' in os.environ:
|
||||||
|
set_level(os.environ['SUGAR_LOGGER_LEVEL'])
|
||||||
|
|
||||||
|
if log_filename:
|
||||||
|
try:
|
||||||
|
log_path = os.path.join(get_logs_dir(), log_filename + '.log')
|
||||||
|
|
||||||
|
log_fd = os.open(log_path, os.O_WRONLY | os.O_CREAT)
|
||||||
|
os.dup2(log_fd, sys.stdout.fileno())
|
||||||
|
os.dup2(log_fd, sys.stderr.fileno())
|
||||||
|
os.close(log_fd)
|
||||||
|
|
||||||
|
sys.stdout = SafeLogWrapper(sys.stdout)
|
||||||
|
sys.stderr = SafeLogWrapper(sys.stderr)
|
||||||
|
except OSError, e:
|
||||||
|
# if we're out of space, just continue
|
||||||
|
if e.errno != errno.ENOSPC:
|
||||||
|
raise e
|
||||||
|
|
||||||
|
sys.excepthook = _except_hook
|
||||||
|
|
||||||
|
|
||||||
|
class TraceRepr(repr_.Repr):
|
||||||
|
|
||||||
|
# better handling of subclasses of basic types, e.g. for DBus
|
||||||
|
_TYPES = [int, long, bool, tuple, list, array.array, set, frozenset,
|
||||||
|
collections.deque, dict, str]
|
||||||
|
|
||||||
|
def repr1(self, x, level):
|
||||||
|
for t in self._TYPES:
|
||||||
|
if isinstance(x, t):
|
||||||
|
return getattr(self, 'repr_'+t.__name__)(x, level)
|
||||||
|
|
||||||
|
return repr_.Repr.repr1(self, x, level)
|
||||||
|
|
||||||
|
def repr_int(self, x, level):
|
||||||
|
return repr(x)
|
||||||
|
|
||||||
|
def repr_bool(self, x, level):
|
||||||
|
return repr(x)
|
||||||
|
|
||||||
|
|
||||||
|
def trace(logger=None, logger_name=None, skip_args=None, skip_kwargs=None,
|
||||||
|
maxsize_list=30, maxsize_dict=30, maxsize_string=300):
|
||||||
|
|
||||||
|
if skip_args is None:
|
||||||
|
skip_args = []
|
||||||
|
|
||||||
|
if skip_kwargs is None:
|
||||||
|
skip_kwargs = []
|
||||||
|
|
||||||
|
# size-limit repr()
|
||||||
|
trace_repr = TraceRepr()
|
||||||
|
trace_repr.maxlist = maxsize_list
|
||||||
|
trace_repr.maxdict = maxsize_dict
|
||||||
|
trace_repr.maxstring = maxsize_string
|
||||||
|
trace_repr.maxother = maxsize_string
|
||||||
|
trace_logger = logger or logging.getLogger(logger_name)
|
||||||
|
|
||||||
|
def _trace(f, *args, **kwargs):
|
||||||
|
# don't do expensive formatting if loglevel TRACE is not enabled
|
||||||
|
enabled = trace_logger.isEnabledFor(TRACE)
|
||||||
|
if not enabled:
|
||||||
|
return f(*args, **kwargs)
|
||||||
|
|
||||||
|
params_formatted = ", ".join(
|
||||||
|
[trace_repr.repr(a)
|
||||||
|
for (idx, a) in enumerate(args) if idx not in skip_args] + \
|
||||||
|
['%s=%s' % (k, trace_repr.repr(v))
|
||||||
|
for (k, v) in kwargs.items() if k not in skip_kwargs])
|
||||||
|
|
||||||
|
trace_logger.log(TRACE, "%s(%s) invoked", f.__name__,
|
||||||
|
params_formatted)
|
||||||
|
|
||||||
|
try:
|
||||||
|
res = f(*args, **kwargs)
|
||||||
|
except:
|
||||||
|
trace_logger.exception("Exception occured in %s", f.__name__)
|
||||||
|
raise
|
||||||
|
|
||||||
|
trace_logger.log(TRACE, "%s(%s) returned %s", f.__name__,
|
||||||
|
params_formatted, trace_repr.repr(res))
|
||||||
|
|
||||||
|
return res
|
||||||
|
|
||||||
|
return decorator.decorator(_trace)
|
268
src/sugar3/mime.py
Normal file
268
src/sugar3/mime.py
Normal file
@ -0,0 +1,268 @@
|
|||||||
|
# Copyright (C) 2006-2007, Red Hat, Inc.
|
||||||
|
# Copyright (C) 2007, One Laptop Per Child
|
||||||
|
#
|
||||||
|
# This library 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; either
|
||||||
|
# version 2 of the License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This library 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
|
||||||
|
# Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public
|
||||||
|
# License along with this library; if not, write to the
|
||||||
|
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
# Boston, MA 02111-1307, USA.
|
||||||
|
|
||||||
|
"""MIME helpers based on freedesktop specification.
|
||||||
|
|
||||||
|
STABLE.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import logging
|
||||||
|
import gettext
|
||||||
|
|
||||||
|
from sugar3 import _sugarbaseext
|
||||||
|
|
||||||
|
_ = lambda msg: gettext.dgettext('sugar-base', msg)
|
||||||
|
|
||||||
|
GENERIC_TYPE_TEXT = 'Text'
|
||||||
|
GENERIC_TYPE_IMAGE = 'Image'
|
||||||
|
GENERIC_TYPE_AUDIO = 'Audio'
|
||||||
|
GENERIC_TYPE_VIDEO = 'Video'
|
||||||
|
GENERIC_TYPE_LINK = 'Link'
|
||||||
|
|
||||||
|
_extensions = {}
|
||||||
|
_globs_timestamps = []
|
||||||
|
_generic_types = [
|
||||||
|
{
|
||||||
|
'id': GENERIC_TYPE_TEXT,
|
||||||
|
'name': _('Text'),
|
||||||
|
'icon': 'text-x-generic',
|
||||||
|
'types': [
|
||||||
|
'text/plain', 'text/rtf', 'application/pdf', 'application/x-pdf',
|
||||||
|
'text/html', 'application/vnd.oasis.opendocument.text',
|
||||||
|
'application/rtf', 'text/rtf', 'application/epub+zip'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'id': GENERIC_TYPE_IMAGE,
|
||||||
|
'name': _('Image'),
|
||||||
|
'icon': 'image-x-generic',
|
||||||
|
'types': ['image/png', 'image/gif', 'image/jpeg'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'id': GENERIC_TYPE_AUDIO,
|
||||||
|
'name': _('Audio'),
|
||||||
|
'icon': 'audio-x-generic',
|
||||||
|
'types': [
|
||||||
|
'audio/ogg', 'audio/x-wav', 'audio/wav', 'audio/x-vorbis+ogg',
|
||||||
|
'audio/x-mpegurl', 'audio/mpegurl', 'audio/mpeg', 'audio/x-scpls'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'id': GENERIC_TYPE_VIDEO,
|
||||||
|
'name': _('Video'),
|
||||||
|
'icon': 'video-x-generic',
|
||||||
|
'types': ['video/ogg', 'application/ogg', 'video/x-theora+ogg'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'id': GENERIC_TYPE_LINK,
|
||||||
|
'name': _('Link'),
|
||||||
|
'icon': 'text-uri-list',
|
||||||
|
'types': ['text/x-moz-url', 'text/uri-list'],
|
||||||
|
}]
|
||||||
|
|
||||||
|
|
||||||
|
class ObjectType(object):
|
||||||
|
|
||||||
|
def __init__(self, type_id, name, icon, mime_types):
|
||||||
|
self.type_id = type_id
|
||||||
|
self.name = name
|
||||||
|
self.icon = icon
|
||||||
|
self.mime_types = mime_types
|
||||||
|
|
||||||
|
|
||||||
|
def get_generic_type(type_id):
|
||||||
|
types = get_all_generic_types()
|
||||||
|
for generic_type in types:
|
||||||
|
if type_id == generic_type.type_id:
|
||||||
|
return generic_type
|
||||||
|
|
||||||
|
|
||||||
|
def get_all_generic_types():
|
||||||
|
types = []
|
||||||
|
for generic_type in _generic_types:
|
||||||
|
object_type = ObjectType(generic_type['id'], generic_type['name'],
|
||||||
|
generic_type['icon'], generic_type['types'])
|
||||||
|
types.append(object_type)
|
||||||
|
return types
|
||||||
|
|
||||||
|
|
||||||
|
def get_for_file(file_name):
|
||||||
|
if file_name.startswith('file://'):
|
||||||
|
file_name = file_name[7:]
|
||||||
|
|
||||||
|
file_name = os.path.realpath(file_name)
|
||||||
|
|
||||||
|
mime_type = _sugarbaseext.get_mime_type_for_file(file_name)
|
||||||
|
if mime_type == 'application/octet-stream':
|
||||||
|
if _file_looks_like_text(file_name):
|
||||||
|
return 'text/plain'
|
||||||
|
else:
|
||||||
|
return 'application/octet-stream'
|
||||||
|
|
||||||
|
return mime_type
|
||||||
|
|
||||||
|
|
||||||
|
def get_from_file_name(file_name):
|
||||||
|
return _sugarbaseext.get_mime_type_from_file_name(file_name)
|
||||||
|
|
||||||
|
|
||||||
|
def get_mime_icon(mime_type):
|
||||||
|
generic_type = _get_generic_type_for_mime(mime_type)
|
||||||
|
if generic_type:
|
||||||
|
return generic_type['icon']
|
||||||
|
|
||||||
|
return mime_type.replace('/', '-')
|
||||||
|
|
||||||
|
|
||||||
|
def get_mime_description(mime_type):
|
||||||
|
generic_type = _get_generic_type_for_mime(mime_type)
|
||||||
|
if generic_type:
|
||||||
|
return generic_type['name']
|
||||||
|
|
||||||
|
import gio
|
||||||
|
return gio.content_type_get_description(mime_type)
|
||||||
|
|
||||||
|
|
||||||
|
def get_mime_parents(mime_type):
|
||||||
|
return _sugarbaseext.list_mime_parents(mime_type)
|
||||||
|
|
||||||
|
|
||||||
|
def get_primary_extension(mime_type):
|
||||||
|
global _extensions
|
||||||
|
global _globs_timestamps
|
||||||
|
|
||||||
|
dirs = []
|
||||||
|
|
||||||
|
if 'XDG_DATA_HOME' in os.environ:
|
||||||
|
dirs.append(os.environ['XDG_DATA_HOME'])
|
||||||
|
else:
|
||||||
|
dirs.append(os.path.expanduser('~/.local/share/'))
|
||||||
|
|
||||||
|
if 'XDG_DATA_DIRS' in os.environ:
|
||||||
|
dirs.extend(os.environ['XDG_DATA_DIRS'].split(':'))
|
||||||
|
else:
|
||||||
|
dirs.extend(['/usr/local/share/', '/usr/share/'])
|
||||||
|
|
||||||
|
timestamps = []
|
||||||
|
globs_path_list = []
|
||||||
|
for f in dirs:
|
||||||
|
globs_path = os.path.join(f, 'mime', 'globs')
|
||||||
|
if os.path.exists(globs_path):
|
||||||
|
mtime = os.stat(globs_path).st_mtime
|
||||||
|
timestamps.append([globs_path, mtime])
|
||||||
|
globs_path_list.append(globs_path)
|
||||||
|
|
||||||
|
if timestamps != _globs_timestamps:
|
||||||
|
# Clear the old extensions list
|
||||||
|
_extensions = {}
|
||||||
|
|
||||||
|
# FIXME Properly support these types in the system. (#4855)
|
||||||
|
_extensions['audio/ogg'] = 'ogg'
|
||||||
|
_extensions['video/ogg'] = 'ogg'
|
||||||
|
|
||||||
|
for globs_path in globs_path_list:
|
||||||
|
globs_file = open(globs_path)
|
||||||
|
for line in globs_file.readlines():
|
||||||
|
line = line.strip()
|
||||||
|
if not line.startswith('#'):
|
||||||
|
line_type, glob = line.split(':')
|
||||||
|
if glob.startswith('*.'):
|
||||||
|
_extensions[line_type] = glob[2:]
|
||||||
|
|
||||||
|
_globs_timestamps = timestamps
|
||||||
|
|
||||||
|
if mime_type in _extensions:
|
||||||
|
return _extensions[mime_type]
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
_MIME_TYPE_BLACK_LIST = [
|
||||||
|
# Target used only between gtk.TextBuffer instances
|
||||||
|
'application/x-gtk-text-buffer-rich-text',
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def choose_most_significant(mime_types):
|
||||||
|
logging.debug('Choosing between %r.', mime_types)
|
||||||
|
if not mime_types:
|
||||||
|
return ''
|
||||||
|
|
||||||
|
if 'text/uri-list' in mime_types:
|
||||||
|
return 'text/uri-list'
|
||||||
|
|
||||||
|
for mime_category in ['image/', 'application/']:
|
||||||
|
for mime_type in mime_types:
|
||||||
|
|
||||||
|
if mime_type.startswith(mime_category) and \
|
||||||
|
mime_type not in _MIME_TYPE_BLACK_LIST:
|
||||||
|
# skip mozilla private types (second component starts with '_'
|
||||||
|
# or ends with '-priv')
|
||||||
|
if mime_type.split('/')[1].startswith('_') or \
|
||||||
|
mime_type.split('/')[1].endswith('-priv'):
|
||||||
|
continue
|
||||||
|
|
||||||
|
# take out the specifier after ';' that mozilla likes to add
|
||||||
|
mime_type = mime_type.split(';')[0]
|
||||||
|
logging.debug('Choosed %r!', mime_type)
|
||||||
|
return mime_type
|
||||||
|
|
||||||
|
if 'text/x-moz-url' in mime_types:
|
||||||
|
logging.debug('Choosed text/x-moz-url!')
|
||||||
|
return 'text/x-moz-url'
|
||||||
|
|
||||||
|
if 'text/html' in mime_types:
|
||||||
|
logging.debug('Choosed text/html!')
|
||||||
|
return 'text/html'
|
||||||
|
|
||||||
|
if 'text/plain' in mime_types:
|
||||||
|
logging.debug('Choosed text/plain!')
|
||||||
|
return 'text/plain'
|
||||||
|
|
||||||
|
logging.debug('Returning first: %r.', mime_types[0])
|
||||||
|
return mime_types[0]
|
||||||
|
|
||||||
|
|
||||||
|
def split_uri_list(uri_list):
|
||||||
|
return _sugarbaseext.uri_list_extract_uris(uri_list)
|
||||||
|
|
||||||
|
|
||||||
|
def _file_looks_like_text(file_name):
|
||||||
|
f = open(file_name, 'r')
|
||||||
|
try:
|
||||||
|
sample = f.read(256)
|
||||||
|
finally:
|
||||||
|
f.close()
|
||||||
|
|
||||||
|
if '\000' in sample:
|
||||||
|
return False
|
||||||
|
|
||||||
|
for encoding in ('ascii', 'latin_1', 'utf_8', 'utf_16'):
|
||||||
|
try:
|
||||||
|
unicode(sample, encoding)
|
||||||
|
return True
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def _get_generic_type_for_mime(mime_type):
|
||||||
|
for generic_type in _generic_types:
|
||||||
|
if mime_type in generic_type['types']:
|
||||||
|
return generic_type
|
||||||
|
return None
|
850
src/sugar3/xdgmime.c
Normal file
850
src/sugar3/xdgmime.c
Normal file
@ -0,0 +1,850 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmime.c: XDG Mime Spec mime resolver. Based on version 0.11 of the spec.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003,2004 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003,2004 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "xdgmime.h"
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
#include "xdgmimeglob.h"
|
||||||
|
#include "xdgmimemagic.h"
|
||||||
|
#include "xdgmimealias.h"
|
||||||
|
#include "xdgmimeparent.h"
|
||||||
|
#include "xdgmimecache.h"
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <assert.h>
|
||||||
|
|
||||||
|
typedef struct XdgDirTimeList XdgDirTimeList;
|
||||||
|
typedef struct XdgCallbackList XdgCallbackList;
|
||||||
|
|
||||||
|
static int need_reread = TRUE;
|
||||||
|
static time_t last_stat_time = 0;
|
||||||
|
|
||||||
|
static XdgGlobHash *global_hash = NULL;
|
||||||
|
static XdgMimeMagic *global_magic = NULL;
|
||||||
|
static XdgAliasList *alias_list = NULL;
|
||||||
|
static XdgParentList *parent_list = NULL;
|
||||||
|
static XdgDirTimeList *dir_time_list = NULL;
|
||||||
|
static XdgCallbackList *callback_list = NULL;
|
||||||
|
|
||||||
|
XdgMimeCache **_caches = NULL;
|
||||||
|
static int n_caches = 0;
|
||||||
|
|
||||||
|
const char xdg_mime_type_unknown[] = "application/octet-stream";
|
||||||
|
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
XDG_CHECKED_UNCHECKED,
|
||||||
|
XDG_CHECKED_VALID,
|
||||||
|
XDG_CHECKED_INVALID
|
||||||
|
};
|
||||||
|
|
||||||
|
struct XdgDirTimeList
|
||||||
|
{
|
||||||
|
time_t mtime;
|
||||||
|
char *directory_name;
|
||||||
|
int checked;
|
||||||
|
XdgDirTimeList *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct XdgCallbackList
|
||||||
|
{
|
||||||
|
XdgCallbackList *next;
|
||||||
|
XdgCallbackList *prev;
|
||||||
|
int callback_id;
|
||||||
|
XdgMimeCallback callback;
|
||||||
|
void *data;
|
||||||
|
XdgMimeDestroy destroy;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Function called by xdg_run_command_on_dirs. If it returns TRUE, further
|
||||||
|
* directories aren't looked at */
|
||||||
|
typedef int (*XdgDirectoryFunc) (const char *directory,
|
||||||
|
void *user_data);
|
||||||
|
|
||||||
|
static XdgDirTimeList *
|
||||||
|
xdg_dir_time_list_new (void)
|
||||||
|
{
|
||||||
|
XdgDirTimeList *retval;
|
||||||
|
|
||||||
|
retval = calloc (1, sizeof (XdgDirTimeList));
|
||||||
|
retval->checked = XDG_CHECKED_UNCHECKED;
|
||||||
|
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
xdg_dir_time_list_free (XdgDirTimeList *list)
|
||||||
|
{
|
||||||
|
XdgDirTimeList *next;
|
||||||
|
|
||||||
|
while (list)
|
||||||
|
{
|
||||||
|
next = list->next;
|
||||||
|
free (list->directory_name);
|
||||||
|
free (list);
|
||||||
|
list = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
xdg_mime_init_from_directory (const char *directory)
|
||||||
|
{
|
||||||
|
char *file_name;
|
||||||
|
struct stat st;
|
||||||
|
XdgDirTimeList *list;
|
||||||
|
|
||||||
|
assert (directory != NULL);
|
||||||
|
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/mime.cache") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/mime.cache");
|
||||||
|
if (stat (file_name, &st) == 0)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _xdg_mime_cache_new_from_file (file_name);
|
||||||
|
|
||||||
|
if (cache != NULL)
|
||||||
|
{
|
||||||
|
list = xdg_dir_time_list_new ();
|
||||||
|
list->directory_name = file_name;
|
||||||
|
list->mtime = st.st_mtime;
|
||||||
|
list->next = dir_time_list;
|
||||||
|
dir_time_list = list;
|
||||||
|
|
||||||
|
_caches = realloc (_caches, sizeof (XdgMimeCache *) * (n_caches + 2));
|
||||||
|
_caches[n_caches] = cache;
|
||||||
|
_caches[n_caches + 1] = NULL;
|
||||||
|
n_caches++;
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free (file_name);
|
||||||
|
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/globs") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/globs");
|
||||||
|
if (stat (file_name, &st) == 0)
|
||||||
|
{
|
||||||
|
_xdg_mime_glob_read_from_file (global_hash, file_name);
|
||||||
|
|
||||||
|
list = xdg_dir_time_list_new ();
|
||||||
|
list->directory_name = file_name;
|
||||||
|
list->mtime = st.st_mtime;
|
||||||
|
list->next = dir_time_list;
|
||||||
|
dir_time_list = list;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
free (file_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/magic") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/magic");
|
||||||
|
if (stat (file_name, &st) == 0)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_read_from_file (global_magic, file_name);
|
||||||
|
|
||||||
|
list = xdg_dir_time_list_new ();
|
||||||
|
list->directory_name = file_name;
|
||||||
|
list->mtime = st.st_mtime;
|
||||||
|
list->next = dir_time_list;
|
||||||
|
dir_time_list = list;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
free (file_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/aliases") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/aliases");
|
||||||
|
_xdg_mime_alias_read_from_file (alias_list, file_name);
|
||||||
|
free (file_name);
|
||||||
|
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/subclasses") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/subclasses");
|
||||||
|
_xdg_mime_parent_read_from_file (parent_list, file_name);
|
||||||
|
free (file_name);
|
||||||
|
|
||||||
|
return FALSE; /* Keep processing */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Runs a command on all the directories in the search path */
|
||||||
|
static void
|
||||||
|
xdg_run_command_on_dirs (XdgDirectoryFunc func,
|
||||||
|
void *user_data)
|
||||||
|
{
|
||||||
|
const char *xdg_data_home;
|
||||||
|
const char *xdg_data_dirs;
|
||||||
|
const char *ptr;
|
||||||
|
|
||||||
|
xdg_data_home = getenv ("XDG_DATA_HOME");
|
||||||
|
if (xdg_data_home)
|
||||||
|
{
|
||||||
|
if ((func) (xdg_data_home, user_data))
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const char *home;
|
||||||
|
|
||||||
|
home = getenv ("HOME");
|
||||||
|
if (home != NULL)
|
||||||
|
{
|
||||||
|
char *guessed_xdg_home;
|
||||||
|
int stop_processing;
|
||||||
|
|
||||||
|
guessed_xdg_home = malloc (strlen (home) + strlen ("/.local/share/") + 1);
|
||||||
|
strcpy (guessed_xdg_home, home);
|
||||||
|
strcat (guessed_xdg_home, "/.local/share/");
|
||||||
|
stop_processing = (func) (guessed_xdg_home, user_data);
|
||||||
|
free (guessed_xdg_home);
|
||||||
|
|
||||||
|
if (stop_processing)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xdg_data_dirs = getenv ("XDG_DATA_DIRS");
|
||||||
|
if (xdg_data_dirs == NULL)
|
||||||
|
xdg_data_dirs = "/usr/local/share/:/usr/share/";
|
||||||
|
|
||||||
|
ptr = xdg_data_dirs;
|
||||||
|
|
||||||
|
while (*ptr != '\000')
|
||||||
|
{
|
||||||
|
const char *end_ptr;
|
||||||
|
char *dir;
|
||||||
|
int len;
|
||||||
|
int stop_processing;
|
||||||
|
|
||||||
|
end_ptr = ptr;
|
||||||
|
while (*end_ptr != ':' && *end_ptr != '\000')
|
||||||
|
end_ptr ++;
|
||||||
|
|
||||||
|
if (end_ptr == ptr)
|
||||||
|
{
|
||||||
|
ptr++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*end_ptr == ':')
|
||||||
|
len = end_ptr - ptr;
|
||||||
|
else
|
||||||
|
len = end_ptr - ptr + 1;
|
||||||
|
dir = malloc (len + 1);
|
||||||
|
strncpy (dir, ptr, len);
|
||||||
|
dir[len] = '\0';
|
||||||
|
stop_processing = (func) (dir, user_data);
|
||||||
|
free (dir);
|
||||||
|
|
||||||
|
if (stop_processing)
|
||||||
|
return;
|
||||||
|
|
||||||
|
ptr = end_ptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Checks file_path to make sure it has the same mtime as last time it was
|
||||||
|
* checked. If it has a different mtime, or if the file doesn't exist, it
|
||||||
|
* returns FALSE.
|
||||||
|
*
|
||||||
|
* FIXME: This doesn't protect against permission changes.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
xdg_check_file (const char *file_path,
|
||||||
|
int *exists)
|
||||||
|
{
|
||||||
|
struct stat st;
|
||||||
|
|
||||||
|
/* If the file exists */
|
||||||
|
if (stat (file_path, &st) == 0)
|
||||||
|
{
|
||||||
|
XdgDirTimeList *list;
|
||||||
|
|
||||||
|
if (exists)
|
||||||
|
*exists = TRUE;
|
||||||
|
|
||||||
|
for (list = dir_time_list; list; list = list->next)
|
||||||
|
{
|
||||||
|
if (! strcmp (list->directory_name, file_path) &&
|
||||||
|
st.st_mtime == list->mtime)
|
||||||
|
{
|
||||||
|
if (list->checked == XDG_CHECKED_UNCHECKED)
|
||||||
|
list->checked = XDG_CHECKED_VALID;
|
||||||
|
else if (list->checked == XDG_CHECKED_VALID)
|
||||||
|
list->checked = XDG_CHECKED_INVALID;
|
||||||
|
|
||||||
|
return (list->checked != XDG_CHECKED_VALID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exists)
|
||||||
|
*exists = FALSE;
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
xdg_check_dir (const char *directory,
|
||||||
|
int *invalid_dir_list)
|
||||||
|
{
|
||||||
|
int invalid, exists;
|
||||||
|
char *file_name;
|
||||||
|
|
||||||
|
assert (directory != NULL);
|
||||||
|
|
||||||
|
/* Check the mime.cache file */
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/mime.cache") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/mime.cache");
|
||||||
|
invalid = xdg_check_file (file_name, &exists);
|
||||||
|
free (file_name);
|
||||||
|
if (invalid)
|
||||||
|
{
|
||||||
|
*invalid_dir_list = TRUE;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
else if (exists)
|
||||||
|
{
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check the globs file */
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/globs") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/globs");
|
||||||
|
invalid = xdg_check_file (file_name, NULL);
|
||||||
|
free (file_name);
|
||||||
|
if (invalid)
|
||||||
|
{
|
||||||
|
*invalid_dir_list = TRUE;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check the magic file */
|
||||||
|
file_name = malloc (strlen (directory) + strlen ("/mime/magic") + 1);
|
||||||
|
strcpy (file_name, directory); strcat (file_name, "/mime/magic");
|
||||||
|
invalid = xdg_check_file (file_name, NULL);
|
||||||
|
free (file_name);
|
||||||
|
if (invalid)
|
||||||
|
{
|
||||||
|
*invalid_dir_list = TRUE;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE; /* Keep processing */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Walks through all the mime files stat()ing them to see if they've changed.
|
||||||
|
* Returns TRUE if they have. */
|
||||||
|
static int
|
||||||
|
xdg_check_dirs (void)
|
||||||
|
{
|
||||||
|
XdgDirTimeList *list;
|
||||||
|
int invalid_dir_list = FALSE;
|
||||||
|
|
||||||
|
for (list = dir_time_list; list; list = list->next)
|
||||||
|
list->checked = XDG_CHECKED_UNCHECKED;
|
||||||
|
|
||||||
|
xdg_run_command_on_dirs ((XdgDirectoryFunc) xdg_check_dir,
|
||||||
|
&invalid_dir_list);
|
||||||
|
|
||||||
|
if (invalid_dir_list)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
for (list = dir_time_list; list; list = list->next)
|
||||||
|
{
|
||||||
|
if (list->checked != XDG_CHECKED_VALID)
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We want to avoid stat()ing on every single mime call, so we only look for
|
||||||
|
* newer files every 5 seconds. This will return TRUE if we need to reread the
|
||||||
|
* mime data from disk.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
xdg_check_time_and_dirs (void)
|
||||||
|
{
|
||||||
|
struct timeval tv;
|
||||||
|
time_t current_time;
|
||||||
|
int retval = FALSE;
|
||||||
|
|
||||||
|
gettimeofday (&tv, NULL);
|
||||||
|
current_time = tv.tv_sec;
|
||||||
|
|
||||||
|
if (current_time >= last_stat_time + 5)
|
||||||
|
{
|
||||||
|
retval = xdg_check_dirs ();
|
||||||
|
last_stat_time = current_time;
|
||||||
|
}
|
||||||
|
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Called in every public function. It reloads the hash function if need be.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
xdg_mime_init (void)
|
||||||
|
{
|
||||||
|
if (xdg_check_time_and_dirs ())
|
||||||
|
{
|
||||||
|
xdg_mime_shutdown ();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (need_reread)
|
||||||
|
{
|
||||||
|
global_hash = _xdg_glob_hash_new ();
|
||||||
|
global_magic = _xdg_mime_magic_new ();
|
||||||
|
alias_list = _xdg_mime_alias_list_new ();
|
||||||
|
parent_list = _xdg_mime_parent_list_new ();
|
||||||
|
|
||||||
|
xdg_run_command_on_dirs ((XdgDirectoryFunc) xdg_mime_init_from_directory,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
need_reread = FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
xdg_mime_get_mime_type_for_data (const void *data,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_get_mime_type_for_data (data, len);
|
||||||
|
|
||||||
|
mime_type = _xdg_mime_magic_lookup_data (global_magic, data, len, NULL, 0);
|
||||||
|
|
||||||
|
if (mime_type)
|
||||||
|
return mime_type;
|
||||||
|
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
xdg_mime_get_mime_type_for_file (const char *file_name,
|
||||||
|
struct stat *statbuf)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
/* currently, only a few globs occur twice, and none
|
||||||
|
* more often, so 5 seems plenty.
|
||||||
|
*/
|
||||||
|
const char *mime_types[5];
|
||||||
|
FILE *file;
|
||||||
|
unsigned char *data;
|
||||||
|
int max_extent;
|
||||||
|
int bytes_read;
|
||||||
|
struct stat buf;
|
||||||
|
const char *base_name;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
if (file_name == NULL)
|
||||||
|
return NULL;
|
||||||
|
if (! _xdg_utf8_validate (file_name))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_get_mime_type_for_file (file_name, statbuf);
|
||||||
|
|
||||||
|
base_name = _xdg_get_base_name (file_name);
|
||||||
|
n = _xdg_glob_hash_lookup_file_name (global_hash, base_name, mime_types, 5);
|
||||||
|
|
||||||
|
if (n == 1)
|
||||||
|
return mime_types[0];
|
||||||
|
|
||||||
|
if (!statbuf)
|
||||||
|
{
|
||||||
|
if (stat (file_name, &buf) != 0)
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
|
||||||
|
statbuf = &buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!S_ISREG (statbuf->st_mode))
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
|
||||||
|
/* FIXME: Need to make sure that max_extent isn't totally broken. This could
|
||||||
|
* be large and need getting from a stream instead of just reading it all
|
||||||
|
* in. */
|
||||||
|
max_extent = _xdg_mime_magic_get_buffer_extents (global_magic);
|
||||||
|
data = malloc (max_extent);
|
||||||
|
if (data == NULL)
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
|
||||||
|
file = fopen (file_name, "r");
|
||||||
|
if (file == NULL)
|
||||||
|
{
|
||||||
|
free (data);
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
bytes_read = fread (data, 1, max_extent, file);
|
||||||
|
if (ferror (file))
|
||||||
|
{
|
||||||
|
free (data);
|
||||||
|
fclose (file);
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
mime_type = _xdg_mime_magic_lookup_data (global_magic, data, bytes_read,
|
||||||
|
mime_types, n);
|
||||||
|
|
||||||
|
free (data);
|
||||||
|
fclose (file);
|
||||||
|
|
||||||
|
if (mime_type)
|
||||||
|
return mime_type;
|
||||||
|
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
xdg_mime_get_mime_type_from_file_name (const char *file_name)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_get_mime_type_from_file_name (file_name);
|
||||||
|
|
||||||
|
if (_xdg_glob_hash_lookup_file_name (global_hash, file_name, &mime_type, 1))
|
||||||
|
return mime_type;
|
||||||
|
else
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xdg_mime_is_valid_mime_type (const char *mime_type)
|
||||||
|
{
|
||||||
|
/* FIXME: We should make this a better test
|
||||||
|
*/
|
||||||
|
return _xdg_utf8_validate (mime_type);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
xdg_mime_shutdown (void)
|
||||||
|
{
|
||||||
|
XdgCallbackList *list;
|
||||||
|
|
||||||
|
/* FIXME: Need to make this (and the whole library) thread safe */
|
||||||
|
if (dir_time_list)
|
||||||
|
{
|
||||||
|
xdg_dir_time_list_free (dir_time_list);
|
||||||
|
dir_time_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (global_hash)
|
||||||
|
{
|
||||||
|
_xdg_glob_hash_free (global_hash);
|
||||||
|
global_hash = NULL;
|
||||||
|
}
|
||||||
|
if (global_magic)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_free (global_magic);
|
||||||
|
global_magic = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (alias_list)
|
||||||
|
{
|
||||||
|
_xdg_mime_alias_list_free (alias_list);
|
||||||
|
alias_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parent_list)
|
||||||
|
{
|
||||||
|
_xdg_mime_parent_list_free (parent_list);
|
||||||
|
parent_list = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < n_caches; i++)
|
||||||
|
_xdg_mime_cache_unref (_caches[i]);
|
||||||
|
free (_caches);
|
||||||
|
_caches = NULL;
|
||||||
|
n_caches = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (list = callback_list; list; list = list->next)
|
||||||
|
(list->callback) (list->data);
|
||||||
|
|
||||||
|
need_reread = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xdg_mime_get_max_buffer_extents (void)
|
||||||
|
{
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_get_max_buffer_extents ();
|
||||||
|
|
||||||
|
return _xdg_mime_magic_get_buffer_extents (global_magic);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_unalias_mime_type (const char *mime_type)
|
||||||
|
{
|
||||||
|
const char *lookup;
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_unalias_mime_type (mime_type);
|
||||||
|
|
||||||
|
if ((lookup = _xdg_mime_alias_list_lookup (alias_list, mime_type)) != NULL)
|
||||||
|
return lookup;
|
||||||
|
|
||||||
|
return mime_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
xdg_mime_unalias_mime_type (const char *mime_type)
|
||||||
|
{
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
return _xdg_mime_unalias_mime_type (mime_type);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_mime_mime_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b)
|
||||||
|
{
|
||||||
|
const char *unalias_a, *unalias_b;
|
||||||
|
|
||||||
|
unalias_a = _xdg_mime_unalias_mime_type (mime_a);
|
||||||
|
unalias_b = _xdg_mime_unalias_mime_type (mime_b);
|
||||||
|
|
||||||
|
if (strcmp (unalias_a, unalias_b) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xdg_mime_mime_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b)
|
||||||
|
{
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
return _xdg_mime_mime_type_equal (mime_a, mime_b);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xdg_mime_media_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b)
|
||||||
|
{
|
||||||
|
char *sep;
|
||||||
|
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
sep = strchr (mime_a, '/');
|
||||||
|
|
||||||
|
if (sep && strncmp (mime_a, mime_b, sep - mime_a + 1) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 1
|
||||||
|
static int
|
||||||
|
xdg_mime_is_super_type (const char *mime)
|
||||||
|
{
|
||||||
|
int length;
|
||||||
|
const char *type;
|
||||||
|
|
||||||
|
length = strlen (mime);
|
||||||
|
type = &(mime[length - 2]);
|
||||||
|
|
||||||
|
if (strcmp (type, "/*") == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_mime_mime_type_subclass (const char *mime,
|
||||||
|
const char *base)
|
||||||
|
{
|
||||||
|
const char *umime, *ubase;
|
||||||
|
const char **parents;
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_mime_type_subclass (mime, base);
|
||||||
|
|
||||||
|
umime = _xdg_mime_unalias_mime_type (mime);
|
||||||
|
ubase = _xdg_mime_unalias_mime_type (base);
|
||||||
|
|
||||||
|
if (strcmp (umime, ubase) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
#if 1
|
||||||
|
/* Handle supertypes */
|
||||||
|
if (xdg_mime_is_super_type (ubase) &&
|
||||||
|
xdg_mime_media_type_equal (umime, ubase))
|
||||||
|
return 1;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Handle special cases text/plain and application/octet-stream */
|
||||||
|
if (strcmp (ubase, "text/plain") == 0 &&
|
||||||
|
strncmp (umime, "text/", 5) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (strcmp (ubase, "application/octet-stream") == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
parents = _xdg_mime_parent_list_lookup (parent_list, umime);
|
||||||
|
for (; parents && *parents; parents++)
|
||||||
|
{
|
||||||
|
if (_xdg_mime_mime_type_subclass (*parents, ubase))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
xdg_mime_mime_type_subclass (const char *mime,
|
||||||
|
const char *base)
|
||||||
|
{
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
return _xdg_mime_mime_type_subclass (mime, base);
|
||||||
|
}
|
||||||
|
|
||||||
|
char **
|
||||||
|
xdg_mime_list_mime_parents (const char *mime)
|
||||||
|
{
|
||||||
|
const char **parents;
|
||||||
|
char **result;
|
||||||
|
int i, n;
|
||||||
|
|
||||||
|
if (_caches)
|
||||||
|
return _xdg_mime_cache_list_mime_parents (mime);
|
||||||
|
|
||||||
|
parents = xdg_mime_get_mime_parents (mime);
|
||||||
|
|
||||||
|
if (!parents)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
for (i = 0; parents[i]; i++) ;
|
||||||
|
|
||||||
|
n = (i + 1) * sizeof (char *);
|
||||||
|
result = (char **) malloc (n);
|
||||||
|
memcpy (result, parents, n);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char **
|
||||||
|
xdg_mime_get_mime_parents (const char *mime)
|
||||||
|
{
|
||||||
|
const char *umime;
|
||||||
|
|
||||||
|
xdg_mime_init ();
|
||||||
|
|
||||||
|
umime = _xdg_mime_unalias_mime_type (mime);
|
||||||
|
|
||||||
|
return _xdg_mime_parent_list_lookup (parent_list, umime);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
xdg_mime_dump (void)
|
||||||
|
{
|
||||||
|
printf ("*** ALIASES ***\n\n");
|
||||||
|
_xdg_mime_alias_list_dump (alias_list);
|
||||||
|
printf ("\n*** PARENTS ***\n\n");
|
||||||
|
_xdg_mime_parent_list_dump (parent_list);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Registers a function to be called every time the mime database reloads its files
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
xdg_mime_register_reload_callback (XdgMimeCallback callback,
|
||||||
|
void *data,
|
||||||
|
XdgMimeDestroy destroy)
|
||||||
|
{
|
||||||
|
XdgCallbackList *list_el;
|
||||||
|
static int callback_id = 1;
|
||||||
|
|
||||||
|
/* Make a new list element */
|
||||||
|
list_el = calloc (1, sizeof (XdgCallbackList));
|
||||||
|
list_el->callback_id = callback_id;
|
||||||
|
list_el->callback = callback;
|
||||||
|
list_el->data = data;
|
||||||
|
list_el->destroy = destroy;
|
||||||
|
list_el->next = callback_list;
|
||||||
|
if (list_el->next)
|
||||||
|
list_el->next->prev = list_el;
|
||||||
|
|
||||||
|
callback_list = list_el;
|
||||||
|
callback_id ++;
|
||||||
|
|
||||||
|
return callback_id - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
xdg_mime_remove_callback (int callback_id)
|
||||||
|
{
|
||||||
|
XdgCallbackList *list;
|
||||||
|
|
||||||
|
for (list = callback_list; list; list = list->next)
|
||||||
|
{
|
||||||
|
if (list->callback_id == callback_id)
|
||||||
|
{
|
||||||
|
if (list->next)
|
||||||
|
list->next = list->prev;
|
||||||
|
|
||||||
|
if (list->prev)
|
||||||
|
list->prev->next = list->next;
|
||||||
|
else
|
||||||
|
callback_list = list->next;
|
||||||
|
|
||||||
|
/* invoke the destroy handler */
|
||||||
|
(list->destroy) (list->data);
|
||||||
|
free (list);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
111
src/sugar3/xdgmime.h
Normal file
111
src/sugar3/xdgmime.h
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmime.h: XDG Mime Spec mime resolver. Based on version 0.11 of the spec.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_H__
|
||||||
|
#define __XDG_MIME_H__
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif /* __cplusplus */
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define XDG_ENTRY(func) _XDG_ENTRY2(XDG_PREFIX,func)
|
||||||
|
#define _XDG_ENTRY2(prefix,func) _XDG_ENTRY3(prefix,func)
|
||||||
|
#define _XDG_ENTRY3(prefix,func) prefix##_##func
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef void (*XdgMimeCallback) (void *user_data);
|
||||||
|
typedef void (*XdgMimeDestroy) (void *user_data);
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define xdg_mime_get_mime_type_for_data XDG_ENTRY(get_mime_type_for_data)
|
||||||
|
#define xdg_mime_get_mime_type_for_file XDG_ENTRY(get_mime_type_for_file)
|
||||||
|
#define xdg_mime_get_mime_type_from_file_name XDG_ENTRY(get_mime_type_from_file_name)
|
||||||
|
#define xdg_mime_is_valid_mime_type XDG_ENTRY(is_valid_mime_type)
|
||||||
|
#define xdg_mime_mime_type_equal XDG_ENTRY(mime_type_equal)
|
||||||
|
#define xdg_mime_media_type_equal XDG_ENTRY(media_type_equal)
|
||||||
|
#define xdg_mime_mime_type_subclass XDG_ENTRY(mime_type_subclass)
|
||||||
|
#define xdg_mime_get_mime_parents XDG_ENTRY(get_mime_parents)
|
||||||
|
#define xdg_mime_list_mime_parents XDG_ENTRY(list_mime_parents)
|
||||||
|
#define xdg_mime_unalias_mime_type XDG_ENTRY(unalias_mime_type)
|
||||||
|
#define xdg_mime_get_max_buffer_extents XDG_ENTRY(get_max_buffer_extents)
|
||||||
|
#define xdg_mime_shutdown XDG_ENTRY(shutdown)
|
||||||
|
#define xdg_mime_dump XDG_ENTRY(dump)
|
||||||
|
#define xdg_mime_register_reload_callback XDG_ENTRY(register_reload_callback)
|
||||||
|
#define xdg_mime_remove_callback XDG_ENTRY(remove_callback)
|
||||||
|
#define xdg_mime_type_unknown XDG_ENTRY(type_unknown)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
extern const char xdg_mime_type_unknown[];
|
||||||
|
#define XDG_MIME_TYPE_UNKNOWN xdg_mime_type_unknown
|
||||||
|
|
||||||
|
const char *xdg_mime_get_mime_type_for_data (const void *data,
|
||||||
|
size_t len);
|
||||||
|
const char *xdg_mime_get_mime_type_for_file (const char *file_name,
|
||||||
|
struct stat *statbuf);
|
||||||
|
const char *xdg_mime_get_mime_type_from_file_name (const char *file_name);
|
||||||
|
int xdg_mime_is_valid_mime_type (const char *mime_type);
|
||||||
|
int xdg_mime_mime_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
int xdg_mime_media_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
int xdg_mime_mime_type_subclass (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
/* xdg_mime_get_mime_parents() is deprecated since it does
|
||||||
|
* not work correctly with caches. Use xdg_mime_list_parents()
|
||||||
|
* instead, but notice that that function expects you to free
|
||||||
|
* the array it returns.
|
||||||
|
*/
|
||||||
|
const char **xdg_mime_get_mime_parents (const char *mime);
|
||||||
|
char ** xdg_mime_list_mime_parents (const char *mime);
|
||||||
|
const char *xdg_mime_unalias_mime_type (const char *mime);
|
||||||
|
int xdg_mime_get_max_buffer_extents (void);
|
||||||
|
void xdg_mime_shutdown (void);
|
||||||
|
void xdg_mime_dump (void);
|
||||||
|
int xdg_mime_register_reload_callback (XdgMimeCallback callback,
|
||||||
|
void *data,
|
||||||
|
XdgMimeDestroy destroy);
|
||||||
|
void xdg_mime_remove_callback (int callback_id);
|
||||||
|
|
||||||
|
/* Private versions of functions that don't call xdg_mime_init () */
|
||||||
|
int _xdg_mime_mime_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
int _xdg_mime_media_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
int _xdg_mime_mime_type_subclass (const char *mime,
|
||||||
|
const char *base);
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif /* __cplusplus */
|
||||||
|
#endif /* __XDG_MIME_H__ */
|
184
src/sugar3/xdgmimealias.c
Normal file
184
src/sugar3/xdgmimealias.c
Normal file
@ -0,0 +1,184 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimealias.c: Private file. Datastructure for storing the aliases.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2004 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2004 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "xdgmimealias.h"
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <assert.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fnmatch.h>
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct XdgAlias XdgAlias;
|
||||||
|
|
||||||
|
struct XdgAlias
|
||||||
|
{
|
||||||
|
char *alias;
|
||||||
|
char *mime_type;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct XdgAliasList
|
||||||
|
{
|
||||||
|
struct XdgAlias *aliases;
|
||||||
|
int n_aliases;
|
||||||
|
};
|
||||||
|
|
||||||
|
XdgAliasList *
|
||||||
|
_xdg_mime_alias_list_new (void)
|
||||||
|
{
|
||||||
|
XdgAliasList *list;
|
||||||
|
|
||||||
|
list = malloc (sizeof (XdgAliasList));
|
||||||
|
|
||||||
|
list->aliases = NULL;
|
||||||
|
list->n_aliases = 0;
|
||||||
|
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_alias_list_free (XdgAliasList *list)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (list->aliases)
|
||||||
|
{
|
||||||
|
for (i = 0; i < list->n_aliases; i++)
|
||||||
|
{
|
||||||
|
free (list->aliases[i].alias);
|
||||||
|
free (list->aliases[i].mime_type);
|
||||||
|
}
|
||||||
|
free (list->aliases);
|
||||||
|
}
|
||||||
|
free (list);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
alias_entry_cmp (const void *v1, const void *v2)
|
||||||
|
{
|
||||||
|
return strcmp (((XdgAlias *)v1)->alias, ((XdgAlias *)v2)->alias);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_alias_list_lookup (XdgAliasList *list,
|
||||||
|
const char *alias)
|
||||||
|
{
|
||||||
|
XdgAlias *entry;
|
||||||
|
XdgAlias key;
|
||||||
|
|
||||||
|
if (list->n_aliases > 0)
|
||||||
|
{
|
||||||
|
key.alias = (char *)alias;
|
||||||
|
key.mime_type = 0;
|
||||||
|
|
||||||
|
entry = bsearch (&key, list->aliases, list->n_aliases,
|
||||||
|
sizeof (XdgAlias), alias_entry_cmp);
|
||||||
|
if (entry)
|
||||||
|
return entry->mime_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_alias_read_from_file (XdgAliasList *list,
|
||||||
|
const char *file_name)
|
||||||
|
{
|
||||||
|
FILE *file;
|
||||||
|
char line[255];
|
||||||
|
int alloc;
|
||||||
|
|
||||||
|
file = fopen (file_name, "r");
|
||||||
|
|
||||||
|
if (file == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* FIXME: Not UTF-8 safe. Doesn't work if lines are greater than 255 chars.
|
||||||
|
* Blah */
|
||||||
|
alloc = list->n_aliases + 16;
|
||||||
|
list->aliases = realloc (list->aliases, alloc * sizeof (XdgAlias));
|
||||||
|
while (fgets (line, 255, file) != NULL)
|
||||||
|
{
|
||||||
|
char *sep;
|
||||||
|
if (line[0] == '#')
|
||||||
|
continue;
|
||||||
|
|
||||||
|
sep = strchr (line, ' ');
|
||||||
|
if (sep == NULL)
|
||||||
|
continue;
|
||||||
|
*(sep++) = '\000';
|
||||||
|
sep[strlen (sep) -1] = '\000';
|
||||||
|
if (list->n_aliases == alloc)
|
||||||
|
{
|
||||||
|
alloc <<= 1;
|
||||||
|
list->aliases = realloc (list->aliases,
|
||||||
|
alloc * sizeof (XdgAlias));
|
||||||
|
}
|
||||||
|
list->aliases[list->n_aliases].alias = strdup (line);
|
||||||
|
list->aliases[list->n_aliases].mime_type = strdup (sep);
|
||||||
|
list->n_aliases++;
|
||||||
|
}
|
||||||
|
list->aliases = realloc (list->aliases,
|
||||||
|
list->n_aliases * sizeof (XdgAlias));
|
||||||
|
|
||||||
|
fclose (file);
|
||||||
|
|
||||||
|
if (list->n_aliases > 1)
|
||||||
|
qsort (list->aliases, list->n_aliases,
|
||||||
|
sizeof (XdgAlias), alias_entry_cmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_alias_list_dump (XdgAliasList *list)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (list->aliases)
|
||||||
|
{
|
||||||
|
for (i = 0; i < list->n_aliases; i++)
|
||||||
|
{
|
||||||
|
printf ("%s %s\n",
|
||||||
|
list->aliases[i].alias,
|
||||||
|
list->aliases[i].mime_type);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
50
src/sugar3/xdgmimealias.h
Normal file
50
src/sugar3/xdgmimealias.h
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimealias.h: Private file. Datastructure for storing the aliases.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2004 Red Hat, Inc.
|
||||||
|
* Copyright (C) 200 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_ALIAS_H__
|
||||||
|
#define __XDG_MIME_ALIAS_H__
|
||||||
|
|
||||||
|
#include "xdgmime.h"
|
||||||
|
|
||||||
|
typedef struct XdgAliasList XdgAliasList;
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define _xdg_mime_alias_read_from_file XDG_ENTRY(alias_read_from_file)
|
||||||
|
#define _xdg_mime_alias_list_new XDG_ENTRY(alias_list_new)
|
||||||
|
#define _xdg_mime_alias_list_free XDG_ENTRY(alias_list_free)
|
||||||
|
#define _xdg_mime_alias_list_lookup XDG_ENTRY(alias_list_lookup)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void _xdg_mime_alias_read_from_file (XdgAliasList *list,
|
||||||
|
const char *file_name);
|
||||||
|
XdgAliasList *_xdg_mime_alias_list_new (void);
|
||||||
|
void _xdg_mime_alias_list_free (XdgAliasList *list);
|
||||||
|
const char *_xdg_mime_alias_list_lookup (XdgAliasList *list,
|
||||||
|
const char *alias);
|
||||||
|
void _xdg_mime_alias_list_dump (XdgAliasList *list);
|
||||||
|
|
||||||
|
#endif /* __XDG_MIME_ALIAS_H__ */
|
895
src/sugar3/xdgmimecache.c
Normal file
895
src/sugar3/xdgmimecache.c
Normal file
@ -0,0 +1,895 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimealias.c: Private file. mmappable caches for mime data
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2005 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <fnmatch.h>
|
||||||
|
#include <assert.h>
|
||||||
|
|
||||||
|
#include <netinet/in.h> /* for ntohl/ntohs */
|
||||||
|
|
||||||
|
#ifdef HAVE_MMAP
|
||||||
|
#include <sys/mman.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#include "xdgmimecache.h"
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
|
||||||
|
#ifndef MAX
|
||||||
|
#define MAX(a,b) ((a) > (b) ? (a) : (b))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef _O_BINARY
|
||||||
|
#define _O_BINARY 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef MAP_FAILED
|
||||||
|
#define MAP_FAILED ((void *) -1)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define MAJOR_VERSION 1
|
||||||
|
#define MINOR_VERSION 0
|
||||||
|
|
||||||
|
struct _XdgMimeCache
|
||||||
|
{
|
||||||
|
int ref_count;
|
||||||
|
|
||||||
|
size_t size;
|
||||||
|
char *buffer;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define GET_UINT16(cache,offset) (ntohs(*(xdg_uint16_t*)((cache) + (offset))))
|
||||||
|
#define GET_UINT32(cache,offset) (ntohl(*(xdg_uint32_t*)((cache) + (offset))))
|
||||||
|
|
||||||
|
XdgMimeCache *
|
||||||
|
_xdg_mime_cache_ref (XdgMimeCache *cache)
|
||||||
|
{
|
||||||
|
cache->ref_count++;
|
||||||
|
return cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_cache_unref (XdgMimeCache *cache)
|
||||||
|
{
|
||||||
|
cache->ref_count--;
|
||||||
|
|
||||||
|
if (cache->ref_count == 0)
|
||||||
|
{
|
||||||
|
#ifdef HAVE_MMAP
|
||||||
|
munmap (cache->buffer, cache->size);
|
||||||
|
#endif
|
||||||
|
free (cache);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
XdgMimeCache *
|
||||||
|
_xdg_mime_cache_new_from_file (const char *file_name)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = NULL;
|
||||||
|
|
||||||
|
#ifdef HAVE_MMAP
|
||||||
|
int fd = -1;
|
||||||
|
struct stat st;
|
||||||
|
char *buffer = NULL;
|
||||||
|
|
||||||
|
/* Open the file and map it into memory */
|
||||||
|
fd = open (file_name, O_RDONLY|_O_BINARY, 0);
|
||||||
|
|
||||||
|
if (fd < 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (fstat (fd, &st) < 0 || st.st_size < 4)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
buffer = (char *) mmap (NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0);
|
||||||
|
|
||||||
|
if (buffer == MAP_FAILED)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
/* Verify version */
|
||||||
|
if (GET_UINT16 (buffer, 0) != MAJOR_VERSION ||
|
||||||
|
GET_UINT16 (buffer, 2) != MINOR_VERSION)
|
||||||
|
{
|
||||||
|
munmap (buffer, st.st_size);
|
||||||
|
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
|
cache = (XdgMimeCache *) malloc (sizeof (XdgMimeCache));
|
||||||
|
cache->ref_count = 1;
|
||||||
|
cache->buffer = buffer;
|
||||||
|
cache->size = st.st_size;
|
||||||
|
|
||||||
|
done:
|
||||||
|
if (fd != -1)
|
||||||
|
close (fd);
|
||||||
|
|
||||||
|
#endif /* HAVE_MMAP */
|
||||||
|
|
||||||
|
return cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_magic_matchlet_compare_to_data (XdgMimeCache *cache,
|
||||||
|
xdg_uint32_t offset,
|
||||||
|
const void *data,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
xdg_uint32_t range_start = GET_UINT32 (cache->buffer, offset);
|
||||||
|
xdg_uint32_t range_length = GET_UINT32 (cache->buffer, offset + 4);
|
||||||
|
xdg_uint32_t data_length = GET_UINT32 (cache->buffer, offset + 12);
|
||||||
|
xdg_uint32_t data_offset = GET_UINT32 (cache->buffer, offset + 16);
|
||||||
|
xdg_uint32_t mask_offset = GET_UINT32 (cache->buffer, offset + 20);
|
||||||
|
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
for (i = range_start; i <= range_start + range_length; i++)
|
||||||
|
{
|
||||||
|
int valid_matchlet = TRUE;
|
||||||
|
|
||||||
|
if (i + data_length > len)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (mask_offset)
|
||||||
|
{
|
||||||
|
for (j = 0; j < data_length; j++)
|
||||||
|
{
|
||||||
|
if ((((unsigned char *)cache->buffer)[data_offset + j] & ((unsigned char *)cache->buffer)[mask_offset + j]) !=
|
||||||
|
((((unsigned char *) data)[j + i]) & ((unsigned char *)cache->buffer)[mask_offset + j]))
|
||||||
|
{
|
||||||
|
valid_matchlet = FALSE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (j = 0; j < data_length; j++)
|
||||||
|
{
|
||||||
|
if (((unsigned char *)cache->buffer)[data_offset + j] != ((unsigned char *) data)[j + i])
|
||||||
|
{
|
||||||
|
valid_matchlet = FALSE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (valid_matchlet)
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_magic_matchlet_compare (XdgMimeCache *cache,
|
||||||
|
xdg_uint32_t offset,
|
||||||
|
const void *data,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
xdg_uint32_t n_children = GET_UINT32 (cache->buffer, offset + 24);
|
||||||
|
xdg_uint32_t child_offset = GET_UINT32 (cache->buffer, offset + 28);
|
||||||
|
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (cache_magic_matchlet_compare_to_data (cache, offset, data, len))
|
||||||
|
{
|
||||||
|
if (n_children == 0)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
for (i = 0; i < n_children; i++)
|
||||||
|
{
|
||||||
|
if (cache_magic_matchlet_compare (cache, child_offset + 32 * i,
|
||||||
|
data, len))
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *
|
||||||
|
cache_magic_compare_to_data (XdgMimeCache *cache,
|
||||||
|
xdg_uint32_t offset,
|
||||||
|
const void *data,
|
||||||
|
size_t len,
|
||||||
|
int *prio)
|
||||||
|
{
|
||||||
|
xdg_uint32_t priority = GET_UINT32 (cache->buffer, offset);
|
||||||
|
xdg_uint32_t mimetype_offset = GET_UINT32 (cache->buffer, offset + 4);
|
||||||
|
xdg_uint32_t n_matchlets = GET_UINT32 (cache->buffer, offset + 8);
|
||||||
|
xdg_uint32_t matchlet_offset = GET_UINT32 (cache->buffer, offset + 12);
|
||||||
|
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < n_matchlets; i++)
|
||||||
|
{
|
||||||
|
if (cache_magic_matchlet_compare (cache, matchlet_offset + i * 32,
|
||||||
|
data, len))
|
||||||
|
{
|
||||||
|
*prio = priority;
|
||||||
|
|
||||||
|
return cache->buffer + mimetype_offset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *
|
||||||
|
cache_magic_lookup_data (XdgMimeCache *cache,
|
||||||
|
const void *data,
|
||||||
|
size_t len,
|
||||||
|
int *prio,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
xdg_uint32_t list_offset;
|
||||||
|
xdg_uint32_t n_entries;
|
||||||
|
xdg_uint32_t offset;
|
||||||
|
|
||||||
|
int j, n;
|
||||||
|
|
||||||
|
*prio = 0;
|
||||||
|
|
||||||
|
list_offset = GET_UINT32 (cache->buffer, 24);
|
||||||
|
n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 8);
|
||||||
|
|
||||||
|
for (j = 0; j < n_entries; j++)
|
||||||
|
{
|
||||||
|
const char *match;
|
||||||
|
|
||||||
|
match = cache_magic_compare_to_data (cache, offset + 16 * j,
|
||||||
|
data, len, prio);
|
||||||
|
if (match)
|
||||||
|
return match;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
xdg_uint32_t mimetype_offset;
|
||||||
|
const char *non_match;
|
||||||
|
|
||||||
|
mimetype_offset = GET_UINT32 (cache->buffer, offset + 16 * j + 4);
|
||||||
|
non_match = cache->buffer + mimetype_offset;
|
||||||
|
|
||||||
|
for (n = 0; n < n_mime_types; n++)
|
||||||
|
{
|
||||||
|
if (mime_types[n] &&
|
||||||
|
xdg_mime_mime_type_equal (mime_types[n], non_match))
|
||||||
|
mime_types[n] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *
|
||||||
|
cache_alias_lookup (const char *alias)
|
||||||
|
{
|
||||||
|
const char *ptr;
|
||||||
|
int i, min, max, mid, cmp;
|
||||||
|
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 4);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
xdg_uint32_t offset;
|
||||||
|
|
||||||
|
min = 0;
|
||||||
|
max = n_entries - 1;
|
||||||
|
while (max >= min)
|
||||||
|
{
|
||||||
|
mid = (min + max) / 2;
|
||||||
|
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * mid);
|
||||||
|
ptr = cache->buffer + offset;
|
||||||
|
cmp = strcmp (ptr, alias);
|
||||||
|
|
||||||
|
if (cmp < 0)
|
||||||
|
min = mid + 1;
|
||||||
|
else if (cmp > 0)
|
||||||
|
max = mid - 1;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * mid + 4);
|
||||||
|
return cache->buffer + offset;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_glob_lookup_literal (const char *file_name,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
const char *ptr;
|
||||||
|
int i, min, max, mid, cmp;
|
||||||
|
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 12);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
xdg_uint32_t offset;
|
||||||
|
|
||||||
|
min = 0;
|
||||||
|
max = n_entries - 1;
|
||||||
|
while (max >= min)
|
||||||
|
{
|
||||||
|
mid = (min + max) / 2;
|
||||||
|
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * mid);
|
||||||
|
ptr = cache->buffer + offset;
|
||||||
|
cmp = strcmp (ptr, file_name);
|
||||||
|
|
||||||
|
if (cmp < 0)
|
||||||
|
min = mid + 1;
|
||||||
|
else if (cmp > 0)
|
||||||
|
max = mid - 1;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * mid + 4);
|
||||||
|
mime_types[0] = (const char *)(cache->buffer + offset);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_glob_lookup_fnmatch (const char *file_name,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
const char *ptr;
|
||||||
|
|
||||||
|
int i, j, n;
|
||||||
|
|
||||||
|
n = 0;
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 20);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
|
||||||
|
for (j = 0; j < n_entries && n < n_mime_types; j++)
|
||||||
|
{
|
||||||
|
xdg_uint32_t offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * j);
|
||||||
|
xdg_uint32_t mimetype_offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * j + 4);
|
||||||
|
ptr = cache->buffer + offset;
|
||||||
|
mime_type = cache->buffer + mimetype_offset;
|
||||||
|
|
||||||
|
/* FIXME: Not UTF-8 safe */
|
||||||
|
if (fnmatch (ptr, file_name, 0) == 0)
|
||||||
|
mime_types[n++] = mime_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_glob_node_lookup_suffix (XdgMimeCache *cache,
|
||||||
|
xdg_uint32_t n_entries,
|
||||||
|
xdg_uint32_t offset,
|
||||||
|
const char *suffix,
|
||||||
|
int ignore_case,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
xdg_unichar_t character;
|
||||||
|
xdg_unichar_t match_char;
|
||||||
|
xdg_uint32_t mimetype_offset;
|
||||||
|
xdg_uint32_t n_children;
|
||||||
|
xdg_uint32_t child_offset;
|
||||||
|
|
||||||
|
int min, max, mid, n, i;
|
||||||
|
|
||||||
|
character = _xdg_utf8_to_ucs4 (suffix);
|
||||||
|
if (ignore_case)
|
||||||
|
character = _xdg_ucs4_to_lower (character);
|
||||||
|
|
||||||
|
min = 0;
|
||||||
|
max = n_entries - 1;
|
||||||
|
while (max >= min)
|
||||||
|
{
|
||||||
|
mid = (min + max) / 2;
|
||||||
|
|
||||||
|
match_char = GET_UINT32 (cache->buffer, offset + 16 * mid);
|
||||||
|
|
||||||
|
if (match_char < character)
|
||||||
|
min = mid + 1;
|
||||||
|
else if (match_char > character)
|
||||||
|
max = mid - 1;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
suffix = _xdg_utf8_next_char (suffix);
|
||||||
|
if (*suffix == '\0')
|
||||||
|
{
|
||||||
|
mimetype_offset = GET_UINT32 (cache->buffer, offset + 16 * mid + 4);
|
||||||
|
n = 0;
|
||||||
|
if (mimetype_offset)
|
||||||
|
mime_types[n++] = cache->buffer + mimetype_offset;
|
||||||
|
|
||||||
|
n_children = GET_UINT32 (cache->buffer, offset + 16 * mid + 8);
|
||||||
|
child_offset = GET_UINT32 (cache->buffer, offset + 16 * mid + 12);
|
||||||
|
i = 0;
|
||||||
|
while (n < n_mime_types && i < n_children)
|
||||||
|
{
|
||||||
|
match_char = GET_UINT32 (cache->buffer, child_offset + 16 * i);
|
||||||
|
mimetype_offset = GET_UINT32 (cache->buffer, offset + 16 * i + 4);
|
||||||
|
if (match_char != 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
mime_types[n++] = cache->buffer + mimetype_offset;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
n_children = GET_UINT32 (cache->buffer, offset + 16 * mid + 8);
|
||||||
|
child_offset = GET_UINT32 (cache->buffer, offset + 16 * mid + 12);
|
||||||
|
|
||||||
|
return cache_glob_node_lookup_suffix (cache,
|
||||||
|
n_children, child_offset,
|
||||||
|
suffix, ignore_case,
|
||||||
|
mime_types,
|
||||||
|
n_mime_types);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_glob_lookup_suffix (const char *suffix,
|
||||||
|
int ignore_case,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
int i, n;
|
||||||
|
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 16);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
xdg_uint32_t offset = GET_UINT32 (cache->buffer, list_offset + 4);
|
||||||
|
|
||||||
|
n = cache_glob_node_lookup_suffix (cache,
|
||||||
|
n_entries, offset,
|
||||||
|
suffix, ignore_case,
|
||||||
|
mime_types,
|
||||||
|
n_mime_types);
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
find_stopchars (char *stopchars)
|
||||||
|
{
|
||||||
|
int i, j, k, l;
|
||||||
|
|
||||||
|
k = 0;
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 16);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
xdg_uint32_t offset = GET_UINT32 (cache->buffer, list_offset + 4);
|
||||||
|
|
||||||
|
for (j = 0; j < n_entries; j++)
|
||||||
|
{
|
||||||
|
xdg_uint32_t match_char = GET_UINT32 (cache->buffer, offset);
|
||||||
|
|
||||||
|
if (match_char < 128)
|
||||||
|
{
|
||||||
|
for (l = 0; l < k; l++)
|
||||||
|
if (stopchars[l] == match_char)
|
||||||
|
break;
|
||||||
|
if (l == k)
|
||||||
|
{
|
||||||
|
stopchars[k] = (char) match_char;
|
||||||
|
k++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
offset += 16;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stopchars[k] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cache_glob_lookup_file_name (const char *file_name,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
const char *ptr;
|
||||||
|
char stopchars[128];
|
||||||
|
int n;
|
||||||
|
|
||||||
|
assert (file_name != NULL);
|
||||||
|
|
||||||
|
/* First, check the literals */
|
||||||
|
n = cache_glob_lookup_literal (file_name, mime_types, n_mime_types);
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
|
||||||
|
find_stopchars (stopchars);
|
||||||
|
|
||||||
|
/* Next, check suffixes */
|
||||||
|
ptr = strpbrk (file_name, stopchars);
|
||||||
|
while (ptr)
|
||||||
|
{
|
||||||
|
n = cache_glob_lookup_suffix (ptr, FALSE, mime_types, n_mime_types);
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
|
||||||
|
n = cache_glob_lookup_suffix (ptr, TRUE, mime_types, n_mime_types);
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
|
||||||
|
ptr = strpbrk (ptr + 1, stopchars);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Last, try fnmatch */
|
||||||
|
return cache_glob_lookup_fnmatch (file_name, mime_types, n_mime_types);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_mime_cache_get_max_buffer_extents (void)
|
||||||
|
{
|
||||||
|
xdg_uint32_t offset;
|
||||||
|
xdg_uint32_t max_extent;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
max_extent = 0;
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
offset = GET_UINT32 (cache->buffer, 24);
|
||||||
|
max_extent = MAX (max_extent, GET_UINT32 (cache->buffer, offset + 4));
|
||||||
|
}
|
||||||
|
|
||||||
|
return max_extent;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *
|
||||||
|
cache_get_mime_type_for_data (const void *data,
|
||||||
|
size_t len,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
int i, n, priority;
|
||||||
|
|
||||||
|
priority = 0;
|
||||||
|
mime_type = NULL;
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
int prio;
|
||||||
|
const char *match;
|
||||||
|
|
||||||
|
match = cache_magic_lookup_data (cache, data, len, &prio,
|
||||||
|
mime_types, n_mime_types);
|
||||||
|
if (prio > priority)
|
||||||
|
{
|
||||||
|
priority = prio;
|
||||||
|
mime_type = match;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (priority > 0)
|
||||||
|
return mime_type;
|
||||||
|
|
||||||
|
for (n = 0; n < n_mime_types; n++)
|
||||||
|
{
|
||||||
|
if (mime_types[n])
|
||||||
|
return mime_types[n];
|
||||||
|
}
|
||||||
|
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_cache_get_mime_type_for_data (const void *data,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
return cache_get_mime_type_for_data (data, len, NULL, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_cache_get_mime_type_for_file (const char *file_name,
|
||||||
|
struct stat *statbuf)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
const char *mime_types[2];
|
||||||
|
FILE *file;
|
||||||
|
unsigned char *data;
|
||||||
|
int max_extent;
|
||||||
|
int bytes_read;
|
||||||
|
struct stat buf;
|
||||||
|
const char *base_name;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
if (file_name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (! _xdg_utf8_validate (file_name))
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
base_name = _xdg_get_base_name (file_name);
|
||||||
|
n = cache_glob_lookup_file_name (base_name, mime_types, 2);
|
||||||
|
|
||||||
|
if (n == 1)
|
||||||
|
return mime_types[0];
|
||||||
|
|
||||||
|
if (!statbuf)
|
||||||
|
{
|
||||||
|
if (stat (file_name, &buf) != 0)
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
|
||||||
|
statbuf = &buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!S_ISREG (statbuf->st_mode))
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
|
||||||
|
/* FIXME: Need to make sure that max_extent isn't totally broken. This could
|
||||||
|
* be large and need getting from a stream instead of just reading it all
|
||||||
|
* in. */
|
||||||
|
max_extent = _xdg_mime_cache_get_max_buffer_extents ();
|
||||||
|
data = malloc (max_extent);
|
||||||
|
if (data == NULL)
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
|
||||||
|
file = fopen (file_name, "r");
|
||||||
|
if (file == NULL)
|
||||||
|
{
|
||||||
|
free (data);
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
bytes_read = fread (data, 1, max_extent, file);
|
||||||
|
if (ferror (file))
|
||||||
|
{
|
||||||
|
free (data);
|
||||||
|
fclose (file);
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
mime_type = cache_get_mime_type_for_data (data, bytes_read,
|
||||||
|
mime_types, n);
|
||||||
|
|
||||||
|
free (data);
|
||||||
|
fclose (file);
|
||||||
|
|
||||||
|
return mime_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_cache_get_mime_type_from_file_name (const char *file_name)
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
|
||||||
|
if (cache_glob_lookup_file_name (file_name, &mime_type, 1))
|
||||||
|
return mime_type;
|
||||||
|
else
|
||||||
|
return XDG_MIME_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 1
|
||||||
|
static int
|
||||||
|
is_super_type (const char *mime)
|
||||||
|
{
|
||||||
|
int length;
|
||||||
|
const char *type;
|
||||||
|
|
||||||
|
length = strlen (mime);
|
||||||
|
type = &(mime[length - 2]);
|
||||||
|
|
||||||
|
if (strcmp (type, "/*") == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_mime_cache_mime_type_subclass (const char *mime,
|
||||||
|
const char *base)
|
||||||
|
{
|
||||||
|
const char *umime, *ubase;
|
||||||
|
|
||||||
|
int i, j, min, max, med, cmp;
|
||||||
|
|
||||||
|
umime = _xdg_mime_cache_unalias_mime_type (mime);
|
||||||
|
ubase = _xdg_mime_cache_unalias_mime_type (base);
|
||||||
|
|
||||||
|
if (strcmp (umime, ubase) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
/* We really want to handle text/ * in GtkFileFilter, so we just
|
||||||
|
* turn on the supertype matching
|
||||||
|
*/
|
||||||
|
#if 1
|
||||||
|
/* Handle supertypes */
|
||||||
|
if (is_super_type (ubase) &&
|
||||||
|
xdg_mime_media_type_equal (umime, ubase))
|
||||||
|
return 1;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Handle special cases text/plain and application/octet-stream */
|
||||||
|
if (strcmp (ubase, "text/plain") == 0 &&
|
||||||
|
strncmp (umime, "text/", 5) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (strcmp (ubase, "application/octet-stream") == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 8);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
xdg_uint32_t offset, n_parents, parent_offset;
|
||||||
|
|
||||||
|
min = 0;
|
||||||
|
max = n_entries - 1;
|
||||||
|
while (max >= min)
|
||||||
|
{
|
||||||
|
med = (min + max)/2;
|
||||||
|
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * med);
|
||||||
|
cmp = strcmp (cache->buffer + offset, umime);
|
||||||
|
if (cmp < 0)
|
||||||
|
min = med + 1;
|
||||||
|
else if (cmp > 0)
|
||||||
|
max = med - 1;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * med + 4);
|
||||||
|
n_parents = GET_UINT32 (cache->buffer, offset);
|
||||||
|
|
||||||
|
for (j = 0; j < n_parents; j++)
|
||||||
|
{
|
||||||
|
parent_offset = GET_UINT32 (cache->buffer, offset + 4 + 4 * j);
|
||||||
|
if (_xdg_mime_cache_mime_type_subclass (cache->buffer + parent_offset, ubase))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_cache_unalias_mime_type (const char *mime)
|
||||||
|
{
|
||||||
|
const char *lookup;
|
||||||
|
|
||||||
|
lookup = cache_alias_lookup (mime);
|
||||||
|
|
||||||
|
if (lookup)
|
||||||
|
return lookup;
|
||||||
|
|
||||||
|
return mime;
|
||||||
|
}
|
||||||
|
|
||||||
|
char **
|
||||||
|
_xdg_mime_cache_list_mime_parents (const char *mime)
|
||||||
|
{
|
||||||
|
int i, j, k, p;
|
||||||
|
char *all_parents[128]; /* we'll stop at 128 */
|
||||||
|
char **result;
|
||||||
|
|
||||||
|
mime = xdg_mime_unalias_mime_type (mime);
|
||||||
|
|
||||||
|
p = 0;
|
||||||
|
for (i = 0; _caches[i]; i++)
|
||||||
|
{
|
||||||
|
XdgMimeCache *cache = _caches[i];
|
||||||
|
|
||||||
|
xdg_uint32_t list_offset = GET_UINT32 (cache->buffer, 8);
|
||||||
|
xdg_uint32_t n_entries = GET_UINT32 (cache->buffer, list_offset);
|
||||||
|
|
||||||
|
for (j = 0; j < n_entries; j++)
|
||||||
|
{
|
||||||
|
xdg_uint32_t mimetype_offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * j);
|
||||||
|
xdg_uint32_t parents_offset = GET_UINT32 (cache->buffer, list_offset + 4 + 8 * j + 4);
|
||||||
|
|
||||||
|
if (strcmp (cache->buffer + mimetype_offset, mime) == 0)
|
||||||
|
{
|
||||||
|
xdg_uint32_t parent_mime_offset;
|
||||||
|
xdg_uint32_t n_parents = GET_UINT32 (cache->buffer, parents_offset);
|
||||||
|
|
||||||
|
for (k = 0; k < n_parents && p < 127; k++)
|
||||||
|
{
|
||||||
|
parent_mime_offset = GET_UINT32 (cache->buffer, parents_offset + 4 + 4 * k);
|
||||||
|
all_parents[p++] = cache->buffer + parent_mime_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
all_parents[p++] = 0;
|
||||||
|
|
||||||
|
result = (char **) malloc (p * sizeof (char *));
|
||||||
|
memcpy (result, all_parents, p * sizeof (char *));
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
63
src/sugar3/xdgmimecache.h
Normal file
63
src/sugar3/xdgmimecache.h
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimecache.h: Private file. Datastructure for mmapped caches.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2005 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_CACHE_H__
|
||||||
|
#define __XDG_MIME_CACHE_H__
|
||||||
|
|
||||||
|
#include "xdgmime.h"
|
||||||
|
|
||||||
|
typedef struct _XdgMimeCache XdgMimeCache;
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define _xdg_mime_cache_new_from_file XDG_ENTRY(cache_new_from_file)
|
||||||
|
#define _xdg_mime_cache_ref XDG_ENTRY(cache_ref)
|
||||||
|
#define _xdg_mime_cache_unref XDG_ENTRY(cache_unref)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
extern XdgMimeCache **_caches;
|
||||||
|
|
||||||
|
XdgMimeCache *_xdg_mime_cache_new_from_file (const char *file_name);
|
||||||
|
XdgMimeCache *_xdg_mime_cache_ref (XdgMimeCache *cache);
|
||||||
|
void _xdg_mime_cache_unref (XdgMimeCache *cache);
|
||||||
|
|
||||||
|
|
||||||
|
const char *_xdg_mime_cache_get_mime_type_for_data (const void *data,
|
||||||
|
size_t len);
|
||||||
|
const char *_xdg_mime_cache_get_mime_type_for_file (const char *file_name,
|
||||||
|
struct stat *statbuf);
|
||||||
|
const char *_xdg_mime_cache_get_mime_type_from_file_name (const char *file_name);
|
||||||
|
int _xdg_mime_cache_is_valid_mime_type (const char *mime_type);
|
||||||
|
int _xdg_mime_cache_mime_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
int _xdg_mime_cache_media_type_equal (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
int _xdg_mime_cache_mime_type_subclass (const char *mime_a,
|
||||||
|
const char *mime_b);
|
||||||
|
char **_xdg_mime_cache_list_mime_parents (const char *mime);
|
||||||
|
const char *_xdg_mime_cache_unalias_mime_type (const char *mime);
|
||||||
|
int _xdg_mime_cache_get_max_buffer_extents (void);
|
||||||
|
|
||||||
|
#endif /* __XDG_MIME_CACHE_H__ */
|
547
src/sugar3/xdgmimeglob.c
Normal file
547
src/sugar3/xdgmimeglob.c
Normal file
@ -0,0 +1,547 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimeglob.c: Private file. Datastructure for storing the globs.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "xdgmimeglob.h"
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <assert.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fnmatch.h>
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct XdgGlobHashNode XdgGlobHashNode;
|
||||||
|
typedef struct XdgGlobList XdgGlobList;
|
||||||
|
|
||||||
|
struct XdgGlobHashNode
|
||||||
|
{
|
||||||
|
xdg_unichar_t character;
|
||||||
|
const char *mime_type;
|
||||||
|
XdgGlobHashNode *next;
|
||||||
|
XdgGlobHashNode *child;
|
||||||
|
};
|
||||||
|
struct XdgGlobList
|
||||||
|
{
|
||||||
|
const char *data;
|
||||||
|
const char *mime_type;
|
||||||
|
XdgGlobList *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct XdgGlobHash
|
||||||
|
{
|
||||||
|
XdgGlobList *literal_list;
|
||||||
|
XdgGlobHashNode *simple_node;
|
||||||
|
XdgGlobList *full_list;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/* XdgGlobList
|
||||||
|
*/
|
||||||
|
static XdgGlobList *
|
||||||
|
_xdg_glob_list_new (void)
|
||||||
|
{
|
||||||
|
XdgGlobList *new_element;
|
||||||
|
|
||||||
|
new_element = calloc (1, sizeof (XdgGlobList));
|
||||||
|
|
||||||
|
return new_element;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Frees glob_list and all of it's children */
|
||||||
|
static void
|
||||||
|
_xdg_glob_list_free (XdgGlobList *glob_list)
|
||||||
|
{
|
||||||
|
XdgGlobList *ptr, *next;
|
||||||
|
|
||||||
|
ptr = glob_list;
|
||||||
|
|
||||||
|
while (ptr != NULL)
|
||||||
|
{
|
||||||
|
next = ptr->next;
|
||||||
|
|
||||||
|
if (ptr->data)
|
||||||
|
free ((void *) ptr->data);
|
||||||
|
if (ptr->mime_type)
|
||||||
|
free ((void *) ptr->mime_type);
|
||||||
|
free (ptr);
|
||||||
|
|
||||||
|
ptr = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static XdgGlobList *
|
||||||
|
_xdg_glob_list_append (XdgGlobList *glob_list,
|
||||||
|
void *data,
|
||||||
|
const char *mime_type)
|
||||||
|
{
|
||||||
|
XdgGlobList *new_element;
|
||||||
|
XdgGlobList *tmp_element;
|
||||||
|
|
||||||
|
new_element = _xdg_glob_list_new ();
|
||||||
|
new_element->data = data;
|
||||||
|
new_element->mime_type = mime_type;
|
||||||
|
if (glob_list == NULL)
|
||||||
|
return new_element;
|
||||||
|
|
||||||
|
tmp_element = glob_list;
|
||||||
|
while (tmp_element->next != NULL)
|
||||||
|
tmp_element = tmp_element->next;
|
||||||
|
|
||||||
|
tmp_element->next = new_element;
|
||||||
|
|
||||||
|
return glob_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
static XdgGlobList *
|
||||||
|
_xdg_glob_list_prepend (XdgGlobList *glob_list,
|
||||||
|
void *data,
|
||||||
|
const char *mime_type)
|
||||||
|
{
|
||||||
|
XdgGlobList *new_element;
|
||||||
|
|
||||||
|
new_element = _xdg_glob_list_new ();
|
||||||
|
new_element->data = data;
|
||||||
|
new_element->next = glob_list;
|
||||||
|
new_element->mime_type = mime_type;
|
||||||
|
|
||||||
|
return new_element;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* XdgGlobHashNode
|
||||||
|
*/
|
||||||
|
|
||||||
|
static XdgGlobHashNode *
|
||||||
|
_xdg_glob_hash_node_new (void)
|
||||||
|
{
|
||||||
|
XdgGlobHashNode *glob_hash_node;
|
||||||
|
|
||||||
|
glob_hash_node = calloc (1, sizeof (XdgGlobHashNode));
|
||||||
|
|
||||||
|
return glob_hash_node;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_xdg_glob_hash_node_dump (XdgGlobHashNode *glob_hash_node,
|
||||||
|
int depth)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < depth; i++)
|
||||||
|
printf (" ");
|
||||||
|
|
||||||
|
printf ("%c", (char)glob_hash_node->character);
|
||||||
|
if (glob_hash_node->mime_type)
|
||||||
|
printf (" - %s\n", glob_hash_node->mime_type);
|
||||||
|
else
|
||||||
|
printf ("\n");
|
||||||
|
if (glob_hash_node->child)
|
||||||
|
_xdg_glob_hash_node_dump (glob_hash_node->child, depth + 1);
|
||||||
|
if (glob_hash_node->next)
|
||||||
|
_xdg_glob_hash_node_dump (glob_hash_node->next, depth);
|
||||||
|
}
|
||||||
|
|
||||||
|
static XdgGlobHashNode *
|
||||||
|
_xdg_glob_hash_insert_text (XdgGlobHashNode *glob_hash_node,
|
||||||
|
const char *text,
|
||||||
|
const char *mime_type)
|
||||||
|
{
|
||||||
|
XdgGlobHashNode *node;
|
||||||
|
xdg_unichar_t character;
|
||||||
|
|
||||||
|
character = _xdg_utf8_to_ucs4 (text);
|
||||||
|
|
||||||
|
if ((glob_hash_node == NULL) ||
|
||||||
|
(character < glob_hash_node->character))
|
||||||
|
{
|
||||||
|
node = _xdg_glob_hash_node_new ();
|
||||||
|
node->character = character;
|
||||||
|
node->next = glob_hash_node;
|
||||||
|
glob_hash_node = node;
|
||||||
|
}
|
||||||
|
else if (character == glob_hash_node->character)
|
||||||
|
{
|
||||||
|
node = glob_hash_node;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
XdgGlobHashNode *prev_node;
|
||||||
|
int found_node = FALSE;
|
||||||
|
|
||||||
|
/* Look for the first character of text in glob_hash_node, and insert it if we
|
||||||
|
* have to.*/
|
||||||
|
prev_node = glob_hash_node;
|
||||||
|
node = prev_node->next;
|
||||||
|
|
||||||
|
while (node != NULL)
|
||||||
|
{
|
||||||
|
if (character < node->character)
|
||||||
|
{
|
||||||
|
node = _xdg_glob_hash_node_new ();
|
||||||
|
node->character = character;
|
||||||
|
node->next = prev_node->next;
|
||||||
|
prev_node->next = node;
|
||||||
|
|
||||||
|
found_node = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else if (character == node->character)
|
||||||
|
{
|
||||||
|
found_node = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
prev_node = node;
|
||||||
|
node = node->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! found_node)
|
||||||
|
{
|
||||||
|
node = _xdg_glob_hash_node_new ();
|
||||||
|
node->character = character;
|
||||||
|
node->next = prev_node->next;
|
||||||
|
prev_node->next = node;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
text = _xdg_utf8_next_char (text);
|
||||||
|
if (*text == '\000')
|
||||||
|
{
|
||||||
|
if (node->mime_type)
|
||||||
|
{
|
||||||
|
if (strcmp (node->mime_type, mime_type))
|
||||||
|
{
|
||||||
|
XdgGlobHashNode *child;
|
||||||
|
int found_node = FALSE;
|
||||||
|
|
||||||
|
child = node->child;
|
||||||
|
while (child && child->character == '\0')
|
||||||
|
{
|
||||||
|
if (strcmp (child->mime_type, mime_type) == 0)
|
||||||
|
{
|
||||||
|
found_node = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
child = child->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!found_node)
|
||||||
|
{
|
||||||
|
child = _xdg_glob_hash_node_new ();
|
||||||
|
child->character = '\000';
|
||||||
|
child->mime_type = mime_type;
|
||||||
|
child->child = NULL;
|
||||||
|
child->next = node->child;
|
||||||
|
node->child = child;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
node->mime_type = mime_type;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
node->child = _xdg_glob_hash_insert_text (node->child, text, mime_type);
|
||||||
|
}
|
||||||
|
return glob_hash_node;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
_xdg_glob_hash_node_lookup_file_name (XdgGlobHashNode *glob_hash_node,
|
||||||
|
const char *file_name,
|
||||||
|
int ignore_case,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
int n;
|
||||||
|
XdgGlobHashNode *node;
|
||||||
|
xdg_unichar_t character;
|
||||||
|
|
||||||
|
if (glob_hash_node == NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
character = _xdg_utf8_to_ucs4 (file_name);
|
||||||
|
if (ignore_case)
|
||||||
|
character = _xdg_ucs4_to_lower(character);
|
||||||
|
|
||||||
|
for (node = glob_hash_node; node && character >= node->character; node = node->next)
|
||||||
|
{
|
||||||
|
if (character == node->character)
|
||||||
|
{
|
||||||
|
file_name = _xdg_utf8_next_char (file_name);
|
||||||
|
if (*file_name == '\000')
|
||||||
|
{
|
||||||
|
n = 0;
|
||||||
|
if (node->mime_type)
|
||||||
|
mime_types[n++] = node->mime_type;
|
||||||
|
node = node->child;
|
||||||
|
while (n < n_mime_types && node && node->character == 0)
|
||||||
|
{
|
||||||
|
if (node->mime_type)
|
||||||
|
mime_types[n++] = node->mime_type;
|
||||||
|
node = node->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
n = _xdg_glob_hash_node_lookup_file_name (node->child,
|
||||||
|
file_name,
|
||||||
|
ignore_case,
|
||||||
|
mime_types,
|
||||||
|
n_mime_types);
|
||||||
|
}
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_glob_hash_lookup_file_name (XdgGlobHash *glob_hash,
|
||||||
|
const char *file_name,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
XdgGlobList *list;
|
||||||
|
const char *ptr;
|
||||||
|
char stopchars[128];
|
||||||
|
int i, n;
|
||||||
|
XdgGlobHashNode *node;
|
||||||
|
|
||||||
|
/* First, check the literals */
|
||||||
|
|
||||||
|
assert (file_name != NULL && n_mime_types > 0);
|
||||||
|
|
||||||
|
for (list = glob_hash->literal_list; list; list = list->next)
|
||||||
|
{
|
||||||
|
if (strcmp ((const char *)list->data, file_name) == 0)
|
||||||
|
{
|
||||||
|
mime_types[0] = list->mime_type;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
i = 0;
|
||||||
|
for (node = glob_hash->simple_node; node; node = node->next)
|
||||||
|
{
|
||||||
|
if (node->character < 128)
|
||||||
|
stopchars[i++] = (char)node->character;
|
||||||
|
}
|
||||||
|
stopchars[i] = '\0';
|
||||||
|
|
||||||
|
ptr = strpbrk (file_name, stopchars);
|
||||||
|
while (ptr)
|
||||||
|
{
|
||||||
|
n = _xdg_glob_hash_node_lookup_file_name (glob_hash->simple_node, ptr, FALSE,
|
||||||
|
mime_types, n_mime_types);
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
|
||||||
|
n = _xdg_glob_hash_node_lookup_file_name (glob_hash->simple_node, ptr, TRUE,
|
||||||
|
mime_types, n_mime_types);
|
||||||
|
if (n > 0)
|
||||||
|
return n;
|
||||||
|
|
||||||
|
ptr = strpbrk (ptr + 1, stopchars);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FIXME: Not UTF-8 safe */
|
||||||
|
n = 0;
|
||||||
|
for (list = glob_hash->full_list; list && n < n_mime_types; list = list->next)
|
||||||
|
{
|
||||||
|
if (fnmatch ((const char *)list->data, file_name, 0) == 0)
|
||||||
|
mime_types[n++] = list->mime_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* XdgGlobHash
|
||||||
|
*/
|
||||||
|
|
||||||
|
XdgGlobHash *
|
||||||
|
_xdg_glob_hash_new (void)
|
||||||
|
{
|
||||||
|
XdgGlobHash *glob_hash;
|
||||||
|
|
||||||
|
glob_hash = calloc (1, sizeof (XdgGlobHash));
|
||||||
|
|
||||||
|
return glob_hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
_xdg_glob_hash_free_nodes (XdgGlobHashNode *node)
|
||||||
|
{
|
||||||
|
if (node)
|
||||||
|
{
|
||||||
|
if (node->child)
|
||||||
|
_xdg_glob_hash_free_nodes (node->child);
|
||||||
|
if (node->next)
|
||||||
|
_xdg_glob_hash_free_nodes (node->next);
|
||||||
|
if (node->mime_type)
|
||||||
|
free ((void *) node->mime_type);
|
||||||
|
free (node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_glob_hash_free (XdgGlobHash *glob_hash)
|
||||||
|
{
|
||||||
|
_xdg_glob_list_free (glob_hash->literal_list);
|
||||||
|
_xdg_glob_list_free (glob_hash->full_list);
|
||||||
|
_xdg_glob_hash_free_nodes (glob_hash->simple_node);
|
||||||
|
free (glob_hash);
|
||||||
|
}
|
||||||
|
|
||||||
|
XdgGlobType
|
||||||
|
_xdg_glob_determine_type (const char *glob)
|
||||||
|
{
|
||||||
|
const char *ptr;
|
||||||
|
int maybe_in_simple_glob = FALSE;
|
||||||
|
int first_char = TRUE;
|
||||||
|
|
||||||
|
ptr = glob;
|
||||||
|
|
||||||
|
while (*ptr != '\000')
|
||||||
|
{
|
||||||
|
if (*ptr == '*' && first_char)
|
||||||
|
maybe_in_simple_glob = TRUE;
|
||||||
|
else if (*ptr == '\\' || *ptr == '[' || *ptr == '?' || *ptr == '*')
|
||||||
|
return XDG_GLOB_FULL;
|
||||||
|
|
||||||
|
first_char = FALSE;
|
||||||
|
ptr = _xdg_utf8_next_char (ptr);
|
||||||
|
}
|
||||||
|
if (maybe_in_simple_glob)
|
||||||
|
return XDG_GLOB_SIMPLE;
|
||||||
|
else
|
||||||
|
return XDG_GLOB_LITERAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* glob must be valid UTF-8 */
|
||||||
|
void
|
||||||
|
_xdg_glob_hash_append_glob (XdgGlobHash *glob_hash,
|
||||||
|
const char *glob,
|
||||||
|
const char *mime_type)
|
||||||
|
{
|
||||||
|
XdgGlobType type;
|
||||||
|
|
||||||
|
assert (glob_hash != NULL);
|
||||||
|
assert (glob != NULL);
|
||||||
|
|
||||||
|
type = _xdg_glob_determine_type (glob);
|
||||||
|
|
||||||
|
switch (type)
|
||||||
|
{
|
||||||
|
case XDG_GLOB_LITERAL:
|
||||||
|
glob_hash->literal_list = _xdg_glob_list_append (glob_hash->literal_list, strdup (glob), strdup (mime_type));
|
||||||
|
break;
|
||||||
|
case XDG_GLOB_SIMPLE:
|
||||||
|
glob_hash->simple_node = _xdg_glob_hash_insert_text (glob_hash->simple_node, glob + 1, strdup (mime_type));
|
||||||
|
break;
|
||||||
|
case XDG_GLOB_FULL:
|
||||||
|
glob_hash->full_list = _xdg_glob_list_append (glob_hash->full_list, strdup (glob), strdup (mime_type));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_glob_hash_dump (XdgGlobHash *glob_hash)
|
||||||
|
{
|
||||||
|
XdgGlobList *list;
|
||||||
|
printf ("LITERAL STRINGS\n");
|
||||||
|
if (glob_hash->literal_list == NULL)
|
||||||
|
{
|
||||||
|
printf (" None\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (list = glob_hash->literal_list; list; list = list->next)
|
||||||
|
printf (" %s - %s\n", (char *)list->data, list->mime_type);
|
||||||
|
}
|
||||||
|
printf ("\nSIMPLE GLOBS\n");
|
||||||
|
_xdg_glob_hash_node_dump (glob_hash->simple_node, 4);
|
||||||
|
|
||||||
|
printf ("\nFULL GLOBS\n");
|
||||||
|
if (glob_hash->full_list == NULL)
|
||||||
|
{
|
||||||
|
printf (" None\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (list = glob_hash->full_list; list; list = list->next)
|
||||||
|
printf (" %s - %s\n", (char *)list->data, list->mime_type);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_glob_read_from_file (XdgGlobHash *glob_hash,
|
||||||
|
const char *file_name)
|
||||||
|
{
|
||||||
|
FILE *glob_file;
|
||||||
|
char line[255];
|
||||||
|
|
||||||
|
glob_file = fopen (file_name, "r");
|
||||||
|
|
||||||
|
if (glob_file == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* FIXME: Not UTF-8 safe. Doesn't work if lines are greater than 255 chars.
|
||||||
|
* Blah */
|
||||||
|
while (fgets (line, 255, glob_file) != NULL)
|
||||||
|
{
|
||||||
|
char *colon;
|
||||||
|
if (line[0] == '#')
|
||||||
|
continue;
|
||||||
|
|
||||||
|
colon = strchr (line, ':');
|
||||||
|
if (colon == NULL)
|
||||||
|
continue;
|
||||||
|
*(colon++) = '\000';
|
||||||
|
colon[strlen (colon) -1] = '\000';
|
||||||
|
_xdg_glob_hash_append_glob (glob_hash, colon, line);
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose (glob_file);
|
||||||
|
}
|
67
src/sugar3/xdgmimeglob.h
Normal file
67
src/sugar3/xdgmimeglob.h
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimeglob.h: Private file. Datastructure for storing the globs.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_GLOB_H__
|
||||||
|
#define __XDG_MIME_GLOB_H__
|
||||||
|
|
||||||
|
#include "xdgmime.h"
|
||||||
|
|
||||||
|
typedef struct XdgGlobHash XdgGlobHash;
|
||||||
|
|
||||||
|
typedef enum
|
||||||
|
{
|
||||||
|
XDG_GLOB_LITERAL, /* Makefile */
|
||||||
|
XDG_GLOB_SIMPLE, /* *.gif */
|
||||||
|
XDG_GLOB_FULL /* x*.[ch] */
|
||||||
|
} XdgGlobType;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define _xdg_mime_glob_read_from_file XDG_ENTRY(glob_read_from_file)
|
||||||
|
#define _xdg_glob_hash_new XDG_ENTRY(hash_new)
|
||||||
|
#define _xdg_glob_hash_free XDG_ENTRY(hash_free)
|
||||||
|
#define _xdg_glob_hash_lookup_file_name XDG_ENTRY(hash_lookup_file_name)
|
||||||
|
#define _xdg_glob_hash_append_glob XDG_ENTRY(hash_append_glob)
|
||||||
|
#define _xdg_glob_determine_type XDG_ENTRY(determine_type)
|
||||||
|
#define _xdg_glob_hash_dump XDG_ENTRY(hash_dump)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void _xdg_mime_glob_read_from_file (XdgGlobHash *glob_hash,
|
||||||
|
const char *file_name);
|
||||||
|
XdgGlobHash *_xdg_glob_hash_new (void);
|
||||||
|
void _xdg_glob_hash_free (XdgGlobHash *glob_hash);
|
||||||
|
int _xdg_glob_hash_lookup_file_name (XdgGlobHash *glob_hash,
|
||||||
|
const char *text,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types);
|
||||||
|
void _xdg_glob_hash_append_glob (XdgGlobHash *glob_hash,
|
||||||
|
const char *glob,
|
||||||
|
const char *mime_type);
|
||||||
|
XdgGlobType _xdg_glob_determine_type (const char *glob);
|
||||||
|
void _xdg_glob_hash_dump (XdgGlobHash *glob_hash);
|
||||||
|
|
||||||
|
#endif /* __XDG_MIME_GLOB_H__ */
|
154
src/sugar3/xdgmimeint.c
Normal file
154
src/sugar3/xdgmimeint.c
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimeint.c: Internal defines and functions.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static const char _xdg_utf8_skip_data[256] = {
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
||||||
|
2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
|
||||||
|
3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,4,4,4,4,4,4,4,4,5,5,5,5,6,6,1,1
|
||||||
|
};
|
||||||
|
|
||||||
|
const char * const _xdg_utf8_skip = _xdg_utf8_skip_data;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Returns the number of unprocessed characters. */
|
||||||
|
xdg_unichar_t
|
||||||
|
_xdg_utf8_to_ucs4(const char *source)
|
||||||
|
{
|
||||||
|
xdg_unichar_t ucs32;
|
||||||
|
if( ! ( *source & 0x80 ) )
|
||||||
|
{
|
||||||
|
ucs32 = *source;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
int bytelength = 0;
|
||||||
|
xdg_unichar_t result;
|
||||||
|
if ( ! (*source & 0x40) )
|
||||||
|
{
|
||||||
|
ucs32 = *source;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ( ! (*source & 0x20) )
|
||||||
|
{
|
||||||
|
result = *source++ & 0x1F;
|
||||||
|
bytelength = 2;
|
||||||
|
}
|
||||||
|
else if ( ! (*source & 0x10) )
|
||||||
|
{
|
||||||
|
result = *source++ & 0x0F;
|
||||||
|
bytelength = 3;
|
||||||
|
}
|
||||||
|
else if ( ! (*source & 0x08) )
|
||||||
|
{
|
||||||
|
result = *source++ & 0x07;
|
||||||
|
bytelength = 4;
|
||||||
|
}
|
||||||
|
else if ( ! (*source & 0x04) )
|
||||||
|
{
|
||||||
|
result = *source++ & 0x03;
|
||||||
|
bytelength = 5;
|
||||||
|
}
|
||||||
|
else if ( ! (*source & 0x02) )
|
||||||
|
{
|
||||||
|
result = *source++ & 0x01;
|
||||||
|
bytelength = 6;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
result = *source++;
|
||||||
|
bytelength = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for ( bytelength --; bytelength > 0; bytelength -- )
|
||||||
|
{
|
||||||
|
result <<= 6;
|
||||||
|
result |= *source++ & 0x3F;
|
||||||
|
}
|
||||||
|
ucs32 = result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ucs32;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* hullo. this is great code. don't rewrite it */
|
||||||
|
|
||||||
|
xdg_unichar_t
|
||||||
|
_xdg_ucs4_to_lower (xdg_unichar_t source)
|
||||||
|
{
|
||||||
|
/* FIXME: Do a real to_upper sometime */
|
||||||
|
/* CaseFolding-3.2.0.txt has a table of rules. */
|
||||||
|
if ((source & 0xFF) == source)
|
||||||
|
return (xdg_unichar_t) tolower ((unsigned char) source);
|
||||||
|
return source;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_utf8_validate (const char *source)
|
||||||
|
{
|
||||||
|
/* FIXME: actually write */
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_get_base_name (const char *file_name)
|
||||||
|
{
|
||||||
|
const char *base_name;
|
||||||
|
|
||||||
|
if (file_name == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
base_name = strrchr (file_name, '/');
|
||||||
|
|
||||||
|
if (base_name == NULL)
|
||||||
|
return file_name;
|
||||||
|
else
|
||||||
|
return base_name + 1;
|
||||||
|
}
|
73
src/sugar3/xdgmimeint.h
Normal file
73
src/sugar3/xdgmimeint.h
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimeint.h: Internal defines and functions.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_INT_H__
|
||||||
|
#define __XDG_MIME_INT_H__
|
||||||
|
|
||||||
|
#include "xdgmime.h"
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* FIXME: Needs to be configure check */
|
||||||
|
typedef unsigned int xdg_unichar_t;
|
||||||
|
typedef unsigned char xdg_uchar8_t;
|
||||||
|
typedef unsigned short xdg_uint16_t;
|
||||||
|
typedef unsigned int xdg_uint32_t;
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define _xdg_utf8_skip XDG_ENTRY(utf8_skip)
|
||||||
|
#define _xdg_utf8_to_ucs4 XDG_ENTRY(utf8_to_ucs4)
|
||||||
|
#define _xdg_ucs4_to_lower XDG_ENTRY(ucs4_to_lower)
|
||||||
|
#define _xdg_utf8_validate XDG_ENTRY(utf8_validate)
|
||||||
|
#define _xdg_get_base_name XDG_ENTRY(get_ase_name)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define SWAP_BE16_TO_LE16(val) (xdg_uint16_t)(((xdg_uint16_t)(val) << 8)|((xdg_uint16_t)(val) >> 8))
|
||||||
|
|
||||||
|
#define SWAP_BE32_TO_LE32(val) (xdg_uint32_t)((((xdg_uint32_t)(val) & 0xFF000000U) >> 24) | \
|
||||||
|
(((xdg_uint32_t)(val) & 0x00FF0000U) >> 8) | \
|
||||||
|
(((xdg_uint32_t)(val) & 0x0000FF00U) << 8) | \
|
||||||
|
(((xdg_uint32_t)(val) & 0x000000FFU) << 24))
|
||||||
|
/* UTF-8 utils
|
||||||
|
*/
|
||||||
|
extern const char *const _xdg_utf8_skip;
|
||||||
|
#define _xdg_utf8_next_char(p) (char *)((p) + _xdg_utf8_skip[*(unsigned char *)(p)])
|
||||||
|
#define _xdg_utf8_char_size(p) (int) (_xdg_utf8_skip[*(unsigned char *)(p)])
|
||||||
|
|
||||||
|
xdg_unichar_t _xdg_utf8_to_ucs4 (const char *source);
|
||||||
|
xdg_unichar_t _xdg_ucs4_to_lower (xdg_unichar_t source);
|
||||||
|
int _xdg_utf8_validate (const char *source);
|
||||||
|
const char *_xdg_get_base_name (const char *file_name);
|
||||||
|
|
||||||
|
#endif /* __XDG_MIME_INT_H__ */
|
807
src/sugar3/xdgmimemagic.c
Normal file
807
src/sugar3/xdgmimemagic.c
Normal file
@ -0,0 +1,807 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimemagic.: Private file. Datastructure for storing magic files.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include "xdgmimemagic.h"
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined getc_unlocked && !defined HAVE_GETC_UNLOCKED
|
||||||
|
# define getc_unlocked(fp) getc (fp)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct XdgMimeMagicMatch XdgMimeMagicMatch;
|
||||||
|
typedef struct XdgMimeMagicMatchlet XdgMimeMagicMatchlet;
|
||||||
|
|
||||||
|
typedef enum
|
||||||
|
{
|
||||||
|
XDG_MIME_MAGIC_SECTION,
|
||||||
|
XDG_MIME_MAGIC_MAGIC,
|
||||||
|
XDG_MIME_MAGIC_ERROR,
|
||||||
|
XDG_MIME_MAGIC_EOF
|
||||||
|
} XdgMimeMagicState;
|
||||||
|
|
||||||
|
struct XdgMimeMagicMatch
|
||||||
|
{
|
||||||
|
const char *mime_type;
|
||||||
|
int priority;
|
||||||
|
XdgMimeMagicMatchlet *matchlet;
|
||||||
|
XdgMimeMagicMatch *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
struct XdgMimeMagicMatchlet
|
||||||
|
{
|
||||||
|
int indent;
|
||||||
|
int offset;
|
||||||
|
unsigned int value_length;
|
||||||
|
unsigned char *value;
|
||||||
|
unsigned char *mask;
|
||||||
|
unsigned int range_length;
|
||||||
|
unsigned int word_size;
|
||||||
|
XdgMimeMagicMatchlet *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
struct XdgMimeMagic
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatch *match_list;
|
||||||
|
int max_extent;
|
||||||
|
};
|
||||||
|
|
||||||
|
static XdgMimeMagicMatch *
|
||||||
|
_xdg_mime_magic_match_new (void)
|
||||||
|
{
|
||||||
|
return calloc (1, sizeof (XdgMimeMagicMatch));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static XdgMimeMagicMatchlet *
|
||||||
|
_xdg_mime_magic_matchlet_new (void)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatchlet *matchlet;
|
||||||
|
|
||||||
|
matchlet = malloc (sizeof (XdgMimeMagicMatchlet));
|
||||||
|
|
||||||
|
matchlet->indent = 0;
|
||||||
|
matchlet->offset = 0;
|
||||||
|
matchlet->value_length = 0;
|
||||||
|
matchlet->value = NULL;
|
||||||
|
matchlet->mask = NULL;
|
||||||
|
matchlet->range_length = 1;
|
||||||
|
matchlet->word_size = 1;
|
||||||
|
matchlet->next = NULL;
|
||||||
|
|
||||||
|
return matchlet;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
_xdg_mime_magic_matchlet_free (XdgMimeMagicMatchlet *mime_magic_matchlet)
|
||||||
|
{
|
||||||
|
if (mime_magic_matchlet)
|
||||||
|
{
|
||||||
|
if (mime_magic_matchlet->next)
|
||||||
|
_xdg_mime_magic_matchlet_free (mime_magic_matchlet->next);
|
||||||
|
if (mime_magic_matchlet->value)
|
||||||
|
free (mime_magic_matchlet->value);
|
||||||
|
if (mime_magic_matchlet->mask)
|
||||||
|
free (mime_magic_matchlet->mask);
|
||||||
|
free (mime_magic_matchlet);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Frees mime_magic_match and the remainder of its list
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
_xdg_mime_magic_match_free (XdgMimeMagicMatch *mime_magic_match)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatch *ptr, *next;
|
||||||
|
|
||||||
|
ptr = mime_magic_match;
|
||||||
|
while (ptr)
|
||||||
|
{
|
||||||
|
next = ptr->next;
|
||||||
|
|
||||||
|
if (ptr->mime_type)
|
||||||
|
free ((void *) ptr->mime_type);
|
||||||
|
if (ptr->matchlet)
|
||||||
|
_xdg_mime_magic_matchlet_free (ptr->matchlet);
|
||||||
|
free (ptr);
|
||||||
|
|
||||||
|
ptr = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Reads in a hunk of data until a newline character or a '\000' is hit. The
|
||||||
|
* returned string is null terminated, and doesn't include the newline.
|
||||||
|
*/
|
||||||
|
static unsigned char *
|
||||||
|
_xdg_mime_magic_read_to_newline (FILE *magic_file,
|
||||||
|
int *end_of_file)
|
||||||
|
{
|
||||||
|
unsigned char *retval;
|
||||||
|
int c;
|
||||||
|
int len, pos;
|
||||||
|
|
||||||
|
len = 128;
|
||||||
|
pos = 0;
|
||||||
|
retval = malloc (len);
|
||||||
|
*end_of_file = FALSE;
|
||||||
|
|
||||||
|
while (TRUE)
|
||||||
|
{
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
{
|
||||||
|
*end_of_file = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (c == '\n' || c == '\000')
|
||||||
|
break;
|
||||||
|
retval[pos++] = (unsigned char) c;
|
||||||
|
if (pos % 128 == 127)
|
||||||
|
{
|
||||||
|
len = len + 128;
|
||||||
|
retval = realloc (retval, len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
retval[pos] = '\000';
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Returns the number read from the file, or -1 if no number could be read.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
_xdg_mime_magic_read_a_number (FILE *magic_file,
|
||||||
|
int *end_of_file)
|
||||||
|
{
|
||||||
|
/* LONG_MAX is about 20 characters on my system */
|
||||||
|
#define MAX_NUMBER_SIZE 30
|
||||||
|
char number_string[MAX_NUMBER_SIZE + 1];
|
||||||
|
int pos = 0;
|
||||||
|
int c;
|
||||||
|
long retval = -1;
|
||||||
|
|
||||||
|
while (TRUE)
|
||||||
|
{
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
|
||||||
|
if (c == EOF)
|
||||||
|
{
|
||||||
|
*end_of_file = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (! isdigit (c))
|
||||||
|
{
|
||||||
|
ungetc (c, magic_file);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
number_string[pos] = (char) c;
|
||||||
|
pos++;
|
||||||
|
if (pos == MAX_NUMBER_SIZE)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (pos > 0)
|
||||||
|
{
|
||||||
|
number_string[pos] = '\000';
|
||||||
|
errno = 0;
|
||||||
|
retval = strtol (number_string, NULL, 10);
|
||||||
|
|
||||||
|
if ((retval < INT_MIN) || (retval > INT_MAX) || (errno != 0))
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Headers are of the format:
|
||||||
|
* [<priority>:<mime-type>]
|
||||||
|
*/
|
||||||
|
static XdgMimeMagicState
|
||||||
|
_xdg_mime_magic_parse_header (FILE *magic_file, XdgMimeMagicMatch *match)
|
||||||
|
{
|
||||||
|
int c;
|
||||||
|
char *buffer;
|
||||||
|
char *end_ptr;
|
||||||
|
int end_of_file = 0;
|
||||||
|
|
||||||
|
assert (magic_file != NULL);
|
||||||
|
assert (match != NULL);
|
||||||
|
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
if (c != '[')
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
|
||||||
|
match->priority = _xdg_mime_magic_read_a_number (magic_file, &end_of_file);
|
||||||
|
if (end_of_file)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
if (match->priority == -1)
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
if (c != ':')
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
|
||||||
|
buffer = (char *)_xdg_mime_magic_read_to_newline (magic_file, &end_of_file);
|
||||||
|
if (end_of_file)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
|
||||||
|
end_ptr = buffer;
|
||||||
|
while (*end_ptr != ']' && *end_ptr != '\000' && *end_ptr != '\n')
|
||||||
|
end_ptr++;
|
||||||
|
if (*end_ptr != ']')
|
||||||
|
{
|
||||||
|
free (buffer);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
*end_ptr = '\000';
|
||||||
|
|
||||||
|
match->mime_type = strdup (buffer);
|
||||||
|
free (buffer);
|
||||||
|
|
||||||
|
return XDG_MIME_MAGIC_MAGIC;
|
||||||
|
}
|
||||||
|
|
||||||
|
static XdgMimeMagicState
|
||||||
|
_xdg_mime_magic_parse_error (FILE *magic_file)
|
||||||
|
{
|
||||||
|
int c;
|
||||||
|
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
if (c == '\n')
|
||||||
|
return XDG_MIME_MAGIC_SECTION;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Headers are of the format:
|
||||||
|
* [ indent ] ">" start-offset "=" value
|
||||||
|
* [ "&" mask ] [ "~" word-size ] [ "+" range-length ] "\n"
|
||||||
|
*/
|
||||||
|
static XdgMimeMagicState
|
||||||
|
_xdg_mime_magic_parse_magic_line (FILE *magic_file,
|
||||||
|
XdgMimeMagicMatch *match)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatchlet *matchlet;
|
||||||
|
int c;
|
||||||
|
int end_of_file;
|
||||||
|
int indent = 0;
|
||||||
|
int bytes_read;
|
||||||
|
|
||||||
|
assert (magic_file != NULL);
|
||||||
|
|
||||||
|
/* Sniff the buffer to make sure it's a valid line */
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
else if (c == '[')
|
||||||
|
{
|
||||||
|
ungetc (c, magic_file);
|
||||||
|
return XDG_MIME_MAGIC_SECTION;
|
||||||
|
}
|
||||||
|
else if (c == '\n')
|
||||||
|
return XDG_MIME_MAGIC_MAGIC;
|
||||||
|
|
||||||
|
/* At this point, it must be a digit or a '>' */
|
||||||
|
end_of_file = FALSE;
|
||||||
|
if (isdigit (c))
|
||||||
|
{
|
||||||
|
ungetc (c, magic_file);
|
||||||
|
indent = _xdg_mime_magic_read_a_number (magic_file, &end_of_file);
|
||||||
|
if (end_of_file)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
if (indent == -1)
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (c != '>')
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
|
||||||
|
matchlet = _xdg_mime_magic_matchlet_new ();
|
||||||
|
matchlet->indent = indent;
|
||||||
|
matchlet->offset = _xdg_mime_magic_read_a_number (magic_file, &end_of_file);
|
||||||
|
if (end_of_file)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
if (matchlet->offset == -1)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
else if (c != '=')
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Next two bytes determine how long the value is */
|
||||||
|
matchlet->value_length = 0;
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
matchlet->value_length = c & 0xFF;
|
||||||
|
matchlet->value_length = matchlet->value_length << 8;
|
||||||
|
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == EOF)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
matchlet->value_length = matchlet->value_length + (c & 0xFF);
|
||||||
|
|
||||||
|
matchlet->value = malloc (matchlet->value_length);
|
||||||
|
|
||||||
|
/* OOM */
|
||||||
|
if (matchlet->value == NULL)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
bytes_read = fread (matchlet->value, 1, matchlet->value_length, magic_file);
|
||||||
|
if (bytes_read != matchlet->value_length)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
if (feof (magic_file))
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
else
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
if (c == '&')
|
||||||
|
{
|
||||||
|
matchlet->mask = malloc (matchlet->value_length);
|
||||||
|
/* OOM */
|
||||||
|
if (matchlet->mask == NULL)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
bytes_read = fread (matchlet->mask, 1, matchlet->value_length, magic_file);
|
||||||
|
if (bytes_read != matchlet->value_length)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
if (feof (magic_file))
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
else
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (c == '~')
|
||||||
|
{
|
||||||
|
matchlet->word_size = _xdg_mime_magic_read_a_number (magic_file, &end_of_file);
|
||||||
|
if (end_of_file)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
if (matchlet->word_size != 0 &&
|
||||||
|
matchlet->word_size != 1 &&
|
||||||
|
matchlet->word_size != 2 &&
|
||||||
|
matchlet->word_size != 4)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (c == '+')
|
||||||
|
{
|
||||||
|
matchlet->range_length = _xdg_mime_magic_read_a_number (magic_file, &end_of_file);
|
||||||
|
if (end_of_file)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
}
|
||||||
|
if (matchlet->range_length == -1)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
c = getc_unlocked (magic_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (c == '\n')
|
||||||
|
{
|
||||||
|
/* We clean up the matchlet, byte swapping if needed */
|
||||||
|
if (matchlet->word_size > 1)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
if (matchlet->value_length % matchlet->word_size != 0)
|
||||||
|
{
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
/* FIXME: need to get this defined in a <config.h> style file */
|
||||||
|
#if LITTLE_ENDIAN
|
||||||
|
for (i = 0; i < matchlet->value_length; i = i + matchlet->word_size)
|
||||||
|
{
|
||||||
|
if (matchlet->word_size == 2)
|
||||||
|
*((xdg_uint16_t *) matchlet->value + i) = SWAP_BE16_TO_LE16 (*((xdg_uint16_t *) (matchlet->value + i)));
|
||||||
|
else if (matchlet->word_size == 4)
|
||||||
|
*((xdg_uint32_t *) matchlet->value + i) = SWAP_BE32_TO_LE32 (*((xdg_uint32_t *) (matchlet->value + i)));
|
||||||
|
if (matchlet->mask)
|
||||||
|
{
|
||||||
|
if (matchlet->word_size == 2)
|
||||||
|
*((xdg_uint16_t *) matchlet->mask + i) = SWAP_BE16_TO_LE16 (*((xdg_uint16_t *) (matchlet->mask + i)));
|
||||||
|
else if (matchlet->word_size == 4)
|
||||||
|
*((xdg_uint32_t *) matchlet->mask + i) = SWAP_BE32_TO_LE32 (*((xdg_uint32_t *) (matchlet->mask + i)));
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
matchlet->next = match->matchlet;
|
||||||
|
match->matchlet = matchlet;
|
||||||
|
|
||||||
|
|
||||||
|
return XDG_MIME_MAGIC_MAGIC;
|
||||||
|
}
|
||||||
|
|
||||||
|
_xdg_mime_magic_matchlet_free (matchlet);
|
||||||
|
if (c == EOF)
|
||||||
|
return XDG_MIME_MAGIC_EOF;
|
||||||
|
|
||||||
|
return XDG_MIME_MAGIC_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
_xdg_mime_magic_matchlet_compare_to_data (XdgMimeMagicMatchlet *matchlet,
|
||||||
|
const void *data,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
int i, j;
|
||||||
|
for (i = matchlet->offset; i < matchlet->offset + matchlet->range_length; i++)
|
||||||
|
{
|
||||||
|
int valid_matchlet = TRUE;
|
||||||
|
|
||||||
|
if (i + matchlet->value_length > len)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (matchlet->mask)
|
||||||
|
{
|
||||||
|
for (j = 0; j < matchlet->value_length; j++)
|
||||||
|
{
|
||||||
|
if ((matchlet->value[j] & matchlet->mask[j]) !=
|
||||||
|
((((unsigned char *) data)[j + i]) & matchlet->mask[j]))
|
||||||
|
{
|
||||||
|
valid_matchlet = FALSE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (j = 0; j < matchlet->value_length; j++)
|
||||||
|
{
|
||||||
|
if (matchlet->value[j] != ((unsigned char *) data)[j + i])
|
||||||
|
{
|
||||||
|
valid_matchlet = FALSE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (valid_matchlet)
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
_xdg_mime_magic_matchlet_compare_level (XdgMimeMagicMatchlet *matchlet,
|
||||||
|
const void *data,
|
||||||
|
size_t len,
|
||||||
|
int indent)
|
||||||
|
{
|
||||||
|
while ((matchlet != NULL) && (matchlet->indent == indent))
|
||||||
|
{
|
||||||
|
if (_xdg_mime_magic_matchlet_compare_to_data (matchlet, data, len))
|
||||||
|
{
|
||||||
|
if ((matchlet->next == NULL) ||
|
||||||
|
(matchlet->next->indent <= indent))
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
if (_xdg_mime_magic_matchlet_compare_level (matchlet->next,
|
||||||
|
data,
|
||||||
|
len,
|
||||||
|
indent + 1))
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
matchlet = matchlet->next;
|
||||||
|
}
|
||||||
|
while (matchlet && matchlet->indent > indent);
|
||||||
|
}
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
_xdg_mime_magic_match_compare_to_data (XdgMimeMagicMatch *match,
|
||||||
|
const void *data,
|
||||||
|
size_t len)
|
||||||
|
{
|
||||||
|
return _xdg_mime_magic_matchlet_compare_level (match->matchlet, data, len, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_xdg_mime_magic_insert_match (XdgMimeMagic *mime_magic,
|
||||||
|
XdgMimeMagicMatch *match)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatch *list;
|
||||||
|
|
||||||
|
if (mime_magic->match_list == NULL)
|
||||||
|
{
|
||||||
|
mime_magic->match_list = match;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (match->priority > mime_magic->match_list->priority)
|
||||||
|
{
|
||||||
|
match->next = mime_magic->match_list;
|
||||||
|
mime_magic->match_list = match;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
list = mime_magic->match_list;
|
||||||
|
while (list->next != NULL)
|
||||||
|
{
|
||||||
|
if (list->next->priority < match->priority)
|
||||||
|
{
|
||||||
|
match->next = list->next;
|
||||||
|
list->next = match;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
list = list->next;
|
||||||
|
}
|
||||||
|
list->next = match;
|
||||||
|
match->next = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
XdgMimeMagic *
|
||||||
|
_xdg_mime_magic_new (void)
|
||||||
|
{
|
||||||
|
return calloc (1, sizeof (XdgMimeMagic));
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_magic_free (XdgMimeMagic *mime_magic)
|
||||||
|
{
|
||||||
|
if (mime_magic) {
|
||||||
|
_xdg_mime_magic_match_free (mime_magic->match_list);
|
||||||
|
free (mime_magic);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_xdg_mime_magic_get_buffer_extents (XdgMimeMagic *mime_magic)
|
||||||
|
{
|
||||||
|
return mime_magic->max_extent;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
_xdg_mime_magic_lookup_data (XdgMimeMagic *mime_magic,
|
||||||
|
const void *data,
|
||||||
|
size_t len,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatch *match;
|
||||||
|
const char *mime_type;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
mime_type = NULL;
|
||||||
|
for (match = mime_magic->match_list; match; match = match->next)
|
||||||
|
{
|
||||||
|
if (_xdg_mime_magic_match_compare_to_data (match, data, len))
|
||||||
|
{
|
||||||
|
if ((mime_type == NULL) || (_xdg_mime_mime_type_subclass (match->mime_type, mime_type))) {
|
||||||
|
mime_type = match->mime_type;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (n = 0; n < n_mime_types; n++)
|
||||||
|
{
|
||||||
|
if (mime_types[n] &&
|
||||||
|
_xdg_mime_mime_type_equal (mime_types[n], match->mime_type))
|
||||||
|
mime_types[n] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mime_type == NULL)
|
||||||
|
{
|
||||||
|
for (n = 0; n < n_mime_types; n++)
|
||||||
|
{
|
||||||
|
if (mime_types[n])
|
||||||
|
mime_type = mime_types[n];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return mime_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_xdg_mime_update_mime_magic_extents (XdgMimeMagic *mime_magic)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatch *match;
|
||||||
|
int max_extent = 0;
|
||||||
|
|
||||||
|
for (match = mime_magic->match_list; match; match = match->next)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatchlet *matchlet;
|
||||||
|
|
||||||
|
for (matchlet = match->matchlet; matchlet; matchlet = matchlet->next)
|
||||||
|
{
|
||||||
|
int extent;
|
||||||
|
|
||||||
|
extent = matchlet->value_length + matchlet->offset + matchlet->range_length;
|
||||||
|
if (max_extent < extent)
|
||||||
|
max_extent = extent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mime_magic->max_extent = max_extent;
|
||||||
|
}
|
||||||
|
|
||||||
|
static XdgMimeMagicMatchlet *
|
||||||
|
_xdg_mime_magic_matchlet_mirror (XdgMimeMagicMatchlet *matchlets)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatchlet *new_list;
|
||||||
|
XdgMimeMagicMatchlet *tmp;
|
||||||
|
|
||||||
|
if ((matchlets == NULL) || (matchlets->next == NULL))
|
||||||
|
return matchlets;
|
||||||
|
|
||||||
|
new_list = NULL;
|
||||||
|
tmp = matchlets;
|
||||||
|
while (tmp != NULL)
|
||||||
|
{
|
||||||
|
XdgMimeMagicMatchlet *matchlet;
|
||||||
|
|
||||||
|
matchlet = tmp;
|
||||||
|
tmp = tmp->next;
|
||||||
|
matchlet->next = new_list;
|
||||||
|
new_list = matchlet;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new_list;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_xdg_mime_magic_read_magic_file (XdgMimeMagic *mime_magic,
|
||||||
|
FILE *magic_file)
|
||||||
|
{
|
||||||
|
XdgMimeMagicState state;
|
||||||
|
XdgMimeMagicMatch *match = NULL; /* Quiet compiler */
|
||||||
|
|
||||||
|
state = XDG_MIME_MAGIC_SECTION;
|
||||||
|
|
||||||
|
while (state != XDG_MIME_MAGIC_EOF)
|
||||||
|
{
|
||||||
|
switch (state)
|
||||||
|
{
|
||||||
|
case XDG_MIME_MAGIC_SECTION:
|
||||||
|
match = _xdg_mime_magic_match_new ();
|
||||||
|
state = _xdg_mime_magic_parse_header (magic_file, match);
|
||||||
|
if (state == XDG_MIME_MAGIC_EOF || state == XDG_MIME_MAGIC_ERROR)
|
||||||
|
_xdg_mime_magic_match_free (match);
|
||||||
|
break;
|
||||||
|
case XDG_MIME_MAGIC_MAGIC:
|
||||||
|
state = _xdg_mime_magic_parse_magic_line (magic_file, match);
|
||||||
|
if (state == XDG_MIME_MAGIC_SECTION ||
|
||||||
|
(state == XDG_MIME_MAGIC_EOF && match->mime_type))
|
||||||
|
{
|
||||||
|
match->matchlet = _xdg_mime_magic_matchlet_mirror (match->matchlet);
|
||||||
|
_xdg_mime_magic_insert_match (mime_magic, match);
|
||||||
|
}
|
||||||
|
else if (state == XDG_MIME_MAGIC_EOF || state == XDG_MIME_MAGIC_ERROR)
|
||||||
|
_xdg_mime_magic_match_free (match);
|
||||||
|
break;
|
||||||
|
case XDG_MIME_MAGIC_ERROR:
|
||||||
|
state = _xdg_mime_magic_parse_error (magic_file);
|
||||||
|
break;
|
||||||
|
case XDG_MIME_MAGIC_EOF:
|
||||||
|
default:
|
||||||
|
/* Make the compiler happy */
|
||||||
|
assert (0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_xdg_mime_update_mime_magic_extents (mime_magic);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_magic_read_from_file (XdgMimeMagic *mime_magic,
|
||||||
|
const char *file_name)
|
||||||
|
{
|
||||||
|
FILE *magic_file;
|
||||||
|
char header[12];
|
||||||
|
|
||||||
|
magic_file = fopen (file_name, "r");
|
||||||
|
|
||||||
|
if (magic_file == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (fread (header, 1, 12, magic_file) == 12)
|
||||||
|
{
|
||||||
|
if (memcmp ("MIME-Magic\0\n", header, 12) == 0)
|
||||||
|
_xdg_mime_magic_read_magic_file (mime_magic, magic_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose (magic_file);
|
||||||
|
}
|
56
src/sugar3/xdgmimemagic.h
Normal file
56
src/sugar3/xdgmimemagic.h
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimemagic.h: Private file. Datastructure for storing the magic files.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2003 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2003 Jonathan Blandford <jrb@alum.mit.edu>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_MAGIC_H__
|
||||||
|
#define __XDG_MIME_MAGIC_H__
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
#include "xdgmime.h"
|
||||||
|
typedef struct XdgMimeMagic XdgMimeMagic;
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define _xdg_mime_glob_read_from_file XDG_ENTRY(glob_read_from_file)
|
||||||
|
#define _xdg_mime_magic_new XDG_ENTRY(magic_new)
|
||||||
|
#define _xdg_mime_magic_read_from_file XDG_ENTRY(magic_read_from_file)
|
||||||
|
#define _xdg_mime_magic_free XDG_ENTRY(magic_free)
|
||||||
|
#define _xdg_mime_magic_get_buffer_extents XDG_ENTRY(magic_get_buffer_extents)
|
||||||
|
#define _xdg_mime_magic_lookup_data XDG_ENTRY(magic_lookup_data)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
XdgMimeMagic *_xdg_mime_magic_new (void);
|
||||||
|
void _xdg_mime_magic_read_from_file (XdgMimeMagic *mime_magic,
|
||||||
|
const char *file_name);
|
||||||
|
void _xdg_mime_magic_free (XdgMimeMagic *mime_magic);
|
||||||
|
int _xdg_mime_magic_get_buffer_extents (XdgMimeMagic *mime_magic);
|
||||||
|
const char *_xdg_mime_magic_lookup_data (XdgMimeMagic *mime_magic,
|
||||||
|
const void *data,
|
||||||
|
size_t len,
|
||||||
|
const char *mime_types[],
|
||||||
|
int n_mime_types);
|
||||||
|
|
||||||
|
#endif /* __XDG_MIME_MAGIC_H__ */
|
219
src/sugar3/xdgmimeparent.c
Normal file
219
src/sugar3/xdgmimeparent.c
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimealias.c: Private file. Datastructure for storing the hierarchy.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2004 Red Hat, Inc.
|
||||||
|
* Copyright (C) 2004 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "xdgmimeparent.h"
|
||||||
|
#include "xdgmimeint.h"
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <assert.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fnmatch.h>
|
||||||
|
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE (!FALSE)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct XdgMimeParents XdgMimeParents;
|
||||||
|
|
||||||
|
struct XdgMimeParents
|
||||||
|
{
|
||||||
|
char *mime;
|
||||||
|
char **parents;
|
||||||
|
int n_parents;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct XdgParentList
|
||||||
|
{
|
||||||
|
struct XdgMimeParents *parents;
|
||||||
|
int n_mimes;
|
||||||
|
};
|
||||||
|
|
||||||
|
XdgParentList *
|
||||||
|
_xdg_mime_parent_list_new (void)
|
||||||
|
{
|
||||||
|
XdgParentList *list;
|
||||||
|
|
||||||
|
list = malloc (sizeof (XdgParentList));
|
||||||
|
|
||||||
|
list->parents = NULL;
|
||||||
|
list->n_mimes = 0;
|
||||||
|
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_parent_list_free (XdgParentList *list)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char **p;
|
||||||
|
|
||||||
|
if (list->parents)
|
||||||
|
{
|
||||||
|
for (i = 0; i < list->n_mimes; i++)
|
||||||
|
{
|
||||||
|
for (p = list->parents[i].parents; *p; p++)
|
||||||
|
free (*p);
|
||||||
|
|
||||||
|
free (list->parents[i].parents);
|
||||||
|
free (list->parents[i].mime);
|
||||||
|
}
|
||||||
|
free (list->parents);
|
||||||
|
}
|
||||||
|
free (list);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
parent_entry_cmp (const void *v1, const void *v2)
|
||||||
|
{
|
||||||
|
return strcmp (((XdgMimeParents *)v1)->mime, ((XdgMimeParents *)v2)->mime);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char **
|
||||||
|
_xdg_mime_parent_list_lookup (XdgParentList *list,
|
||||||
|
const char *mime)
|
||||||
|
{
|
||||||
|
XdgMimeParents *entry;
|
||||||
|
XdgMimeParents key;
|
||||||
|
|
||||||
|
if (list->n_mimes > 0)
|
||||||
|
{
|
||||||
|
key.mime = (char *)mime;
|
||||||
|
key.parents = NULL;
|
||||||
|
|
||||||
|
entry = bsearch (&key, list->parents, list->n_mimes,
|
||||||
|
sizeof (XdgMimeParents), &parent_entry_cmp);
|
||||||
|
if (entry)
|
||||||
|
return (const char **)entry->parents;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_parent_read_from_file (XdgParentList *list,
|
||||||
|
const char *file_name)
|
||||||
|
{
|
||||||
|
FILE *file;
|
||||||
|
char line[255];
|
||||||
|
int i, alloc;
|
||||||
|
XdgMimeParents *entry;
|
||||||
|
|
||||||
|
file = fopen (file_name, "r");
|
||||||
|
|
||||||
|
if (file == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* FIXME: Not UTF-8 safe. Doesn't work if lines are greater than 255 chars.
|
||||||
|
* Blah */
|
||||||
|
alloc = list->n_mimes + 16;
|
||||||
|
list->parents = realloc (list->parents, alloc * sizeof (XdgMimeParents));
|
||||||
|
while (fgets (line, 255, file) != NULL)
|
||||||
|
{
|
||||||
|
char *sep;
|
||||||
|
if (line[0] == '#')
|
||||||
|
continue;
|
||||||
|
|
||||||
|
sep = strchr (line, ' ');
|
||||||
|
if (sep == NULL)
|
||||||
|
continue;
|
||||||
|
*(sep++) = '\000';
|
||||||
|
sep[strlen (sep) -1] = '\000';
|
||||||
|
entry = NULL;
|
||||||
|
for (i = 0; i < list->n_mimes; i++)
|
||||||
|
{
|
||||||
|
if (strcmp (list->parents[i].mime, line) == 0)
|
||||||
|
{
|
||||||
|
entry = &(list->parents[i]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!entry)
|
||||||
|
{
|
||||||
|
if (list->n_mimes == alloc)
|
||||||
|
{
|
||||||
|
alloc <<= 1;
|
||||||
|
list->parents = realloc (list->parents,
|
||||||
|
alloc * sizeof (XdgMimeParents));
|
||||||
|
}
|
||||||
|
list->parents[list->n_mimes].mime = strdup (line);
|
||||||
|
list->parents[list->n_mimes].parents = NULL;
|
||||||
|
entry = &(list->parents[list->n_mimes]);
|
||||||
|
list->n_mimes++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!entry->parents)
|
||||||
|
{
|
||||||
|
entry->n_parents = 1;
|
||||||
|
entry->parents = malloc ((entry->n_parents + 1) * sizeof (char *));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
entry->n_parents += 1;
|
||||||
|
entry->parents = realloc (entry->parents,
|
||||||
|
(entry->n_parents + 2) * sizeof (char *));
|
||||||
|
}
|
||||||
|
entry->parents[entry->n_parents - 1] = strdup (sep);
|
||||||
|
entry->parents[entry->n_parents] = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
list->parents = realloc (list->parents,
|
||||||
|
list->n_mimes * sizeof (XdgMimeParents));
|
||||||
|
|
||||||
|
fclose (file);
|
||||||
|
|
||||||
|
if (list->n_mimes > 1)
|
||||||
|
qsort (list->parents, list->n_mimes,
|
||||||
|
sizeof (XdgMimeParents), &parent_entry_cmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
_xdg_mime_parent_list_dump (XdgParentList *list)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char **p;
|
||||||
|
|
||||||
|
if (list->parents)
|
||||||
|
{
|
||||||
|
for (i = 0; i < list->n_mimes; i++)
|
||||||
|
{
|
||||||
|
for (p = list->parents[i].parents; *p; p++)
|
||||||
|
printf ("%s %s\n", list->parents[i].mime, *p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
50
src/sugar3/xdgmimeparent.h
Normal file
50
src/sugar3/xdgmimeparent.h
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
/* -*- mode: C; c-file-style: "gnu" -*- */
|
||||||
|
/* xdgmimeparent.h: Private file. Datastructure for storing the hierarchy.
|
||||||
|
*
|
||||||
|
* More info can be found at http://www.freedesktop.org/standards/
|
||||||
|
*
|
||||||
|
* Copyright (C) 2004 Red Hat, Inc.
|
||||||
|
* Copyright (C) 200 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Academic Free License version 2.0
|
||||||
|
* Or under the following terms:
|
||||||
|
*
|
||||||
|
* This library 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; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library 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
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __XDG_MIME_PARENT_H__
|
||||||
|
#define __XDG_MIME_PARENT_H__
|
||||||
|
|
||||||
|
#include "xdgmime.h"
|
||||||
|
|
||||||
|
typedef struct XdgParentList XdgParentList;
|
||||||
|
|
||||||
|
#ifdef XDG_PREFIX
|
||||||
|
#define _xdg_mime_parent_read_from_file XDG_ENTRY(parent_read_from_file)
|
||||||
|
#define _xdg_mime_parent_list_new XDG_ENTRY(parent_list_new)
|
||||||
|
#define _xdg_mime_parent_list_free XDG_ENTRY(parent_list_free)
|
||||||
|
#define _xdg_mime_parent_list_lookup XDG_ENTRY(parent_list_lookup)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void _xdg_mime_parent_read_from_file (XdgParentList *list,
|
||||||
|
const char *file_name);
|
||||||
|
XdgParentList *_xdg_mime_parent_list_new (void);
|
||||||
|
void _xdg_mime_parent_list_free (XdgParentList *list);
|
||||||
|
const char **_xdg_mime_parent_list_lookup (XdgParentList *list,
|
||||||
|
const char *mime);
|
||||||
|
void _xdg_mime_parent_list_dump (XdgParentList *list);
|
||||||
|
|
||||||
|
#endif /* __XDG_MIME_PARENT_H__ */
|
Loading…
Reference in New Issue
Block a user