2012-02-22 21:55:59 +01:00
|
|
|
"""
|
|
|
|
This file is part of notmuch.
|
|
|
|
|
|
|
|
Notmuch is free software: you can redistribute it and/or modify it
|
|
|
|
under the terms of the GNU General Public License as published by the
|
|
|
|
Free Software Foundation, either version 3 of the License, or (at your
|
|
|
|
option) any later version.
|
|
|
|
|
|
|
|
Notmuch is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2016-06-02 18:26:14 +02:00
|
|
|
along with notmuch. If not, see <https://www.gnu.org/licenses/>.
|
2012-02-22 21:55:59 +01:00
|
|
|
|
2012-04-29 16:33:06 +02:00
|
|
|
Copyright 2010 Sebastian Spaeth <Sebastian@SSpaeth.de>
|
2012-02-22 21:55:59 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
from ctypes import c_uint, c_long
|
2012-05-17 16:58:53 +02:00
|
|
|
from .globals import (
|
2012-02-22 21:55:59 +01:00
|
|
|
nmlib,
|
2012-02-23 00:11:22 +01:00
|
|
|
NotmuchDirectoryP,
|
|
|
|
NotmuchFilenamesP
|
|
|
|
)
|
|
|
|
from .errors import (
|
2012-02-22 21:55:59 +01:00
|
|
|
STATUS,
|
|
|
|
NotmuchError,
|
|
|
|
NotInitializedError,
|
|
|
|
)
|
2012-02-22 22:44:35 +01:00
|
|
|
from .filenames import Filenames
|
2012-02-22 21:55:59 +01:00
|
|
|
|
|
|
|
class Directory(object):
|
|
|
|
"""Represents a directory entry in the notmuch directory
|
|
|
|
|
|
|
|
Modifying attributes of this object will modify the
|
|
|
|
database, not the real directory attributes.
|
|
|
|
|
|
|
|
The Directory object is usually derived from another object
|
|
|
|
e.g. via :meth:`Database.get_directory`, and will automatically be
|
|
|
|
become invalid whenever that parent is deleted. You should
|
|
|
|
therefore initialized this object handing it a reference to the
|
|
|
|
parent, preventing the parent from automatically being garbage
|
|
|
|
collected.
|
|
|
|
"""
|
|
|
|
|
|
|
|
"""notmuch_directory_get_mtime"""
|
|
|
|
_get_mtime = nmlib.notmuch_directory_get_mtime
|
|
|
|
_get_mtime.argtypes = [NotmuchDirectoryP]
|
|
|
|
_get_mtime.restype = c_long
|
|
|
|
|
|
|
|
"""notmuch_directory_set_mtime"""
|
|
|
|
_set_mtime = nmlib.notmuch_directory_set_mtime
|
|
|
|
_set_mtime.argtypes = [NotmuchDirectoryP, c_long]
|
|
|
|
_set_mtime.restype = c_uint
|
|
|
|
|
|
|
|
"""notmuch_directory_get_child_files"""
|
|
|
|
_get_child_files = nmlib.notmuch_directory_get_child_files
|
|
|
|
_get_child_files.argtypes = [NotmuchDirectoryP]
|
|
|
|
_get_child_files.restype = NotmuchFilenamesP
|
|
|
|
|
|
|
|
"""notmuch_directory_get_child_directories"""
|
|
|
|
_get_child_directories = nmlib.notmuch_directory_get_child_directories
|
|
|
|
_get_child_directories.argtypes = [NotmuchDirectoryP]
|
|
|
|
_get_child_directories.restype = NotmuchFilenamesP
|
|
|
|
|
|
|
|
def _assert_dir_is_initialized(self):
|
|
|
|
"""Raises a NotmuchError(:attr:`STATUS`.NOT_INITIALIZED)
|
|
|
|
if dir_p is None"""
|
|
|
|
if not self._dir_p:
|
|
|
|
raise NotInitializedError()
|
|
|
|
|
|
|
|
def __init__(self, path, dir_p, parent):
|
|
|
|
"""
|
|
|
|
:param path: The absolute path of the directory object.
|
|
|
|
:param dir_p: The pointer to an internal notmuch_directory_t object.
|
|
|
|
:param parent: The object this Directory is derived from
|
|
|
|
(usually a :class:`Database`). We do not directly use
|
|
|
|
this, but store a reference to it as long as
|
|
|
|
this Directory object lives. This keeps the
|
|
|
|
parent object alive.
|
|
|
|
"""
|
|
|
|
self._path = path
|
|
|
|
self._dir_p = dir_p
|
|
|
|
self._parent = parent
|
|
|
|
|
|
|
|
def set_mtime(self, mtime):
|
|
|
|
"""Sets the mtime value of this directory in the database
|
|
|
|
|
|
|
|
The intention is for the caller to use the mtime to allow efficient
|
|
|
|
identification of new messages to be added to the database. The
|
|
|
|
recommended usage is as follows:
|
|
|
|
|
|
|
|
* Read the mtime of a directory from the filesystem
|
|
|
|
|
database: add n_d_index_file (deprecates n_d_add_message)
We need a way to pass parameters to the indexing functionality on the
first index, not just on reindexing. The obvious place is in
notmuch_database_add_message. But since modifying the argument list
would break both API and ABI, we needed a new name.
I considered notmuch_database_add_message_with_params(), but the
functionality we're talking about doesn't always add a message. It
tries to index a specific file, possibly adding a message, but
possibly doing other things, like adding terms to an existing message,
or failing to deal with message objects entirely (e.g. because the
file didn't contain a message).
So i chose the function name notmuch_database_index_file.
I confess i'm a little concerned about confusing future notmuch
developers with the new name, since we already have a private
_notmuch_message_index_file function, and the two do rather different
things. But i think the added clarity for people linking against the
future libnotmuch and the capacity for using index parameters makes
this a worthwhile tradeoff. (that said, if anyone has another name
that they strongly prefer, i'd be happy to go with it)
This changeset also adjusts the tests so that we test whether the new,
preferred function returns bad values (since the deprecated function
just calls the new one).
We can keep the deprecated n_d_add_message function around as long as
we like, but at the next place where we're forced to break API or ABI
we can probably choose to drop the name relatively safely.
NOTE: there is probably more cleanup to do in the ruby and go bindings
to complete the deprecation directly. I don't know those languages
well enough to attempt a fix; i don't know how to test them; and i
don't know the culture around those languages about API additions or
deprecations.
2017-08-18 01:14:25 +02:00
|
|
|
* Call :meth:`Database.index_file` for all mail files in
|
2012-02-22 21:55:59 +01:00
|
|
|
the directory
|
|
|
|
|
|
|
|
* Call notmuch_directory_set_mtime with the mtime read from the
|
|
|
|
filesystem. Then, when wanting to check for updates to the
|
|
|
|
directory in the future, the client can call :meth:`get_mtime`
|
|
|
|
and know that it only needs to add files if the mtime of the
|
|
|
|
directory and files are newer than the stored timestamp.
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
:meth:`get_mtime` function does not allow the caller to
|
|
|
|
distinguish a timestamp of 0 from a non-existent timestamp. So
|
|
|
|
don't store a timestamp of 0 unless you are comfortable with
|
|
|
|
that.
|
|
|
|
|
|
|
|
:param mtime: A (time_t) timestamp
|
|
|
|
:raises: :exc:`XapianError` a Xapian exception occurred, mtime
|
|
|
|
not stored
|
|
|
|
:raises: :exc:`ReadOnlyDatabaseError` the database was opened
|
|
|
|
in read-only mode so directory mtime cannot be modified
|
|
|
|
:raises: :exc:`NotInitializedError` the directory object has not
|
|
|
|
been initialized
|
|
|
|
"""
|
|
|
|
self._assert_dir_is_initialized()
|
|
|
|
status = Directory._set_mtime(self._dir_p, mtime)
|
|
|
|
|
|
|
|
if status != STATUS.SUCCESS:
|
|
|
|
raise NotmuchError(status)
|
|
|
|
|
|
|
|
def get_mtime(self):
|
|
|
|
"""Gets the mtime value of this directory in the database
|
|
|
|
|
|
|
|
Retrieves a previously stored mtime for this directory.
|
|
|
|
|
|
|
|
:param mtime: A (time_t) timestamp
|
|
|
|
:raises: :exc:`NotmuchError`:
|
|
|
|
|
|
|
|
:attr:`STATUS`.NOT_INITIALIZED
|
|
|
|
The directory has not been initialized
|
|
|
|
"""
|
|
|
|
self._assert_dir_is_initialized()
|
|
|
|
return Directory._get_mtime(self._dir_p)
|
|
|
|
|
|
|
|
# Make mtime attribute a property of Directory()
|
|
|
|
mtime = property(get_mtime, set_mtime, doc="""Property that allows getting
|
|
|
|
and setting of the Directory *mtime* (read-write)
|
|
|
|
|
|
|
|
See :meth:`get_mtime` and :meth:`set_mtime` for usage and
|
|
|
|
possible exceptions.""")
|
|
|
|
|
|
|
|
def get_child_files(self):
|
|
|
|
"""Gets a Filenames iterator listing all the filenames of
|
|
|
|
messages in the database within the given directory.
|
|
|
|
|
|
|
|
The returned filenames will be the basename-entries only (not
|
|
|
|
complete paths.
|
|
|
|
"""
|
|
|
|
self._assert_dir_is_initialized()
|
|
|
|
files_p = Directory._get_child_files(self._dir_p)
|
|
|
|
return Filenames(files_p, self)
|
|
|
|
|
|
|
|
def get_child_directories(self):
|
|
|
|
"""Gets a :class:`Filenames` iterator listing all the filenames of
|
|
|
|
sub-directories in the database within the given directory
|
|
|
|
|
|
|
|
The returned filenames will be the basename-entries only (not
|
|
|
|
complete paths.
|
|
|
|
"""
|
|
|
|
self._assert_dir_is_initialized()
|
|
|
|
files_p = Directory._get_child_directories(self._dir_p)
|
|
|
|
return Filenames(files_p, self)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def path(self):
|
|
|
|
"""Returns the absolute path of this Directory (read-only)"""
|
|
|
|
return self._path
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
"""Object representation"""
|
|
|
|
return "<notmuch Directory object '%s'>" % self._path
|
|
|
|
|
|
|
|
_destroy = nmlib.notmuch_directory_destroy
|
|
|
|
_destroy.argtypes = [NotmuchDirectoryP]
|
2012-03-16 13:56:32 +01:00
|
|
|
_destroy.restype = None
|
2012-02-22 21:55:59 +01:00
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
"""Close and free the Directory"""
|
2012-04-30 18:52:35 +02:00
|
|
|
if self._dir_p:
|
2012-02-22 21:55:59 +01:00
|
|
|
self._destroy(self._dir_p)
|