File: main_page.dox.in

package info (click to toggle)
persistent-cache-cpp 1.0.7-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 648 kB
  • sloc: cpp: 7,754; python: 183; ansic: 91; sh: 34; makefile: 7
file content (139 lines) | stat: -rw-r--r-- 5,798 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
/*!
\mainpage

\section overview Overview

This API provides a persistent cache of key-value pairs. core::PersistentStringCache stores key,
value, and metadata as type `std::string`. A simple type adapter template, core::PersistentCache,
is provided to permit the use of types other than `string`.

The cache is intended for caching arbitrary (possibly large) amounts of data,
such as might be needed by a web browser cache. The cache scales to large
numbers (millions) of entries and is [very fast](@ref performance).
The implementation is based on [leveldb](http://leveldb.org) and typically
provides throughput many times larger than the I/O bandwidth to disk.

The cache is robust in the face of crashes and power loss. After a
re-start, it is guaranteed to be in a consistent state with correct
data. (Some number of updates that were made just prior to a power loss
or kernel crash can be lost; however, if just the calling process
crashes, no updates are lost.)

A cache has a maximum size (which can be changed at any time). Once
the cache reaches its maximum size, when adding an entry,
the cache automatically discards enough entries to make room for the new entry.

Keys can be (possibly binary) strings of size > 0. Values can be
(possibly binary) strings including the empty string. Using the
core::PersistentCache template, keys, values, and [metadata](@ref metadata)
can be arbitrary user-defined types. The template requires the
the application to provide encode and decode functions that
convert each user-defined type to/from `string` and calls
these functions automatically. This relieves the application
from explicitly having to serialize or deserialize user-defined
types when manipulating the cache.

Entries maintain an access time, which is used to keep them in
least-recently-used (LRU) order. In addition, entries can have
an optional expiry time. (If no expiry time is specified, infinite
expiry time is assumed.)

\note The cache is thread-safe; you can call member functions from
different threads without any synchronization. Thread-safety is
provided for convenience, not performance. Calling concurrently
into the cache from multiple threads will not yield improved performance.

\subsection policy Discard policy

The cache provides two different discard policies, `lru_ttl`
and `lru_only`.

For `lru_ttl`, the discard policy of the cache is to first delete all
entries that have expired. If this does not free sufficient space
to make room for a new entry, the cache then deletes entries in oldest
to newest (LRU) order until sufficient space is available. This
deletion in LRU order may delete entries that have an
expiry time, but have not expired yet, as well as entries with
infinite expiry time.

For `lru_only`, entries do not maintain an expiry time and
are therefore discarded strictly in LRU order.

Access and expiry times are recorded with millisecond granularity.
To indicate infinite expiry time, use the defaulted parameter value or
`chrono::system_clock::time_point()`.

\subsection metadata Metadata

Besides storing key-value pairs, the cache allows you to add arbitrary
extra data to each entry. This is useful, for example, to maintain
metadata (such as HTTP header details) for the entries in the cache.

\warning It is not possible to distinguish between "no metadata was added"
and "empty metadata was added". Do not use the metadata in such
a way that you rely the difference between "metadata not there" and
"metadata is the empty string".

\subsection errors Error reporting

Methods throw `std::runtime_error` if the underlying database
(leveldb) reports an error. If leveldb detects database corruption, the code
throws `std::system_error` with with a 666 error code. To recover
from this error, remove all files in the cache directory.
Other errors are indicated by throwing `std::logic_error` or
`std::invalid_argument` as appropriate.

\subsection performance Performance

Some rough performance figures, taken on an Intel Ivy Bridge i7-3770K 3.5 GHz
with 16 GB RAM, appear below. Records are filled with random data to make them non-compressible.

After filling the cache, the code performs cache lookups using random
keys, with an 80% hit probability. On a miss, it inserts a new
random record. This measures the typical steady-state behavior:
whenever a cache miss happens, the caller fetches the data and
inserts a new record into the cache.

Setting     | Value
----------  | ------
Cache size  | 100 MB
# Records   | ~5100
Record size | 20 kB, normal distribution, stddev = 7000

Running the test With a 7200 rpm spinning disk produces:

Parameter    | Value
-------------|------------
Reads        | 30.9 MB/sec
Writes       | 7.0 MB/sec
Records/sec  | 1995

Running the test With an Intel 256 GB SSD produces:

Parameter    | Value
-------------|------------
Reads        | 80.4 MB/sec
Writes       | 15.7 MB/sec
Records/sec  | 4932

\note When benchmarking, make sure to compile in release mode. In debug
mode, a number of expensive assertions are turned on.

\note Also be aware that leveldb uses Snappy compression beneath the
covers. This means that, if test data is simply filled with
a fixed byte pattern, you will measure artificially high performance.

\subsection linking Compiling and linking

The API is provided as a static library, `lib@LIBNAME@.a`. (Code size on a 64-bit processor is less than 100 kB.)
You can compile and link with the library as follows:

    g++ --std=c++11 -o myprog myprog.cpp `pkg-config --cflags --libs libpersistent-cache-cpp` -lleveldb

\subsection examples Examples

See core::PersistentStringCache and core::PersistentCache for simple usage examples.

Additional code examples can be found in
[@LIBNAME@/examples](file:///@CMAKE_INSTALL_PREFIX@/share/doc/@LIBNAME@/examples).
*/