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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
|
<!doctype article PUBLIC "-//Davenport//DTD DocBook V3.0//EN" [
<!entity Evolution "<application>Evolution</application>">
<!entity Camel "Camel">
]>
<article class="whitepaper" id="camel">
<artheader>
<title>The &Camel; Messaging Library</title>
<authorgroup>
<author>
<firstname>Dan</firstname>
<surname>Winship</surname>
<affiliation>
<address>
<email>danw@helixcode.com</email>
</address>
</affiliation>
</author>
<author>
<firstname>Bertrand</firstname>
<surname>Guiheneuf</surname>
<affiliation>
<address>
<email>bertrand@helixcode.com</email>
</address>
</affiliation>
</author>
</authorgroup>
<copyright>
<year>2000</year>
<holder>Helix Code, Inc.</holder>
</copyright>
</artheader>
<sect1 id="introduction">
<title>Introduction</title>
<para>
&Camel; is a generic messaging library. It is being used as the
back end for the mail component of &Evolution;. The name
"&Camel;" is an acronym; it refers to the fact that the
library is capable of going several days without food or water.
It means : Camel has an Acronym that Makes Everyone Laugh.
</para>
<para>
&Camel;'s initial design is heavily based on Sun's
<trademark>JavaMail</trademark> API. It uses the Gtk+ object
system, and many of its classes are direct analags of JavaMail
classes. Its design has also been influenced by the features of
IMAP, and the limitations of the standard UNIX mbox mail store,
which set some of the boundaries on its requirements and
extensibility.
</para>
<para>
&Camel; sees all message repositories as stores containing
folders. These folders in turn contain the messages the client
actually accesses. The use of such a unified interface allows
the client applications to be very extensible. &Camel; includes
an external provider mechanism which allows applications to
dynamically load and use protocols which were not available when
the application was initially written.
</para>
<para>
The abstract store/folder mechanism is a powerful and versatile
way of accessing messages. No particular asumptions are made on
the client side, thus allowing new ways of managing the
messages. For example, the messages stored in the folders don't
necessarily have to share some common physical location. The
folder can be a purely virtual folder, containing only
references to the actual messages. This is used by the "vFolder"
provider, which allows you select messages meeting particular
criteria and deal with them as a group.
</para>
<para>
In addition to these possibilities, &Camel; has full MIME
support. &Camel; MIME messages are lightweight objects
representing the MIME skeleton of the actual message. The data
contained in the subparts are never stored in memory except when
they are actually needed. The application, when accessing the
various MIME objects contained in the message (text parts,
attachments, embedded binary objects ...) asks &Camel; for a
stream that it can read data from. This scheme is particularly
useful with the IMAP provider. IMAP has strong MIME support
built-in, which allows &Camel; to download only the parts of
messages that it actually needs: attachments need not be
downloaded until they are viewed, and unnecessary
"multipart/alternative" parts will never be read off the server.
</para>
</sect1>
<sect1 id="overview">
<title>Overview</title>
<graphic format="gif" fileref="camel"></graphic>
<para>
To begin using &Camel;, an application first creates a
<classname>CamelSession</classname> object. This object is used
to store application defaults, and to coordinate communication
between providers and the application.
</para>
<para>
A <classname>CamelProvider</classname> is a dynamically-loadable
module that provides functionality associated with a specific
service. Examples of providers are IMAP and SMTP. Providers
include subclasses of the various other &Camel; classes for
accessing and manipulating messages.
</para>
<para>
<classname>CamelService</classname> is an abstract class for
describing a connection to a local or remote service. It
currently has two subclasses: <classname>CamelStore</classname>,
for services that store messages (such as IMAP servers and mbox
files), and <classname>CamelTransport</classname>, for services
that deliver messages (such as SMTP, or a local MTA). A provider
could also be both a store and a transport, as in the case of
NNTP.
</para>
<para>
A <classname>CamelStore</classname> contains some number of
<classname>CamelFolder</classname> objects, which in turn
contain messages. A <classname>CamelFolder</classname> provides
a <classname>CamelFolderSummary</classname> object, which
includes details about the subject, date, and sender of each
message in the folder. The folder also includes the messages
themselves, as subclasses of <classname>CamelMedium</classname>.
</para>
<para>
Email messages are represented by the
<classname>CamelMimeMessage</classname> class, a subclass of
<classname>CamelMedium</classname>. This class includes
operations for accessing RFC822 and MIME headers, accessing
subparts of MIME messages, encoding and decoding Base64 and
Quoted-Printable, etc.
</para>
<para>
<classname>CamelTransport</classname> includes methods for
delivering messages. While the abstract
<function>CamelTransport::send</function> method takes a
<classname>CamelMedium</classname>, its subclasses may only be
able to deliver messages of specific
<classname>CamelMedium</classname> subclasses. For instance,
<classname>CamelSendmailTransport</classname> requires a
<classname>CamelMimeMessage</classname>, because it needs a
message that includes a "To:" header. A hypothetical
<classname>CamelNNTPTransport</classname> would need a
<classname>CamelNewsMessage</classname>, which would have a
"Newsgroups:" header.
</para>
<para>
The content of messages are referred to using
<classname>CamelStream</classname> and its subclasses. In the
case of an mbox-based store, the
<classname>CamelStream</classname> would abstract the operation
of reading the correct section of the mbox file. For IMAP,
reading off the <classname>CamelStream</classname> might result
in commands being issued to the remote IMAP server and data
being read off a socket.
</para>
<para>
The final major class in &Camel; is
<classname>CamelException</classname>, which is used to
propagate information about errors. Many methods take a
<classname>CamelException</classname> as an argument, which the
caller can then check if an error occurs. It includes both a
numeric error code which can be interpreted by the program, and
a text error message that can be displayed to the user.
</para>
</sect1>
<sect1 id="classes">
<title>Major Subcomponents</title>
<sect2 id="store">
<title>The Message Store</title>
<para>
A <classname>CamelStore</classname> inherits the ability to
connect and authenticate to a service from its parent class,
<classname>CamelService</classname>. It then adds the ability
to retrieve folders. A store must contain at least one folder,
which can be retrieved with
<function>CamelStore::get_default_folder</function>. There are
also methods to retrieve the "top-level" folder (for
hieararchical stores), and to retrieve an arbitrary folder by
name.
</para>
<para>
All <classname>CamelFolder</classname>s must implement certain
core operations, most notably generating a summary and
retrieving and deleting messages. A
<classname>CamelFolder</classname> must assign a permanently
unique identifier to each message it contains. Messages can
then be retrieved via
<function>CamelFolder::get_message_by_uid</function>. Alternately,
within a single mail-reading session, messages can be referred
to by their linear position within the store using
<function>CamelFolder::get_message_by_number</function>.
</para>
<para>
Folders must also implement the
<function>get_parent_folder</function> and
<function>list_subfolders</function> methods. For stores that
don't allow multiple folders, they would return NULL and an
empty list, respectively. Stores that do allow multiple
folders will also define methods for creating and deleting
folders, and for moving messages between them (assuming the
folders are writable).
</para>
<para>
Folders that support searching can define the
<function>search_by_expression</function> method. For mbox
folders, this is implemented by indexing the messages with the
ibex library and using that to search them later. For IMAP
folders, this uses the IMAP SEARCH command. Other folder types
might not be able to implement this functionality, in which
case users would not be able to do full-content searches on
them.
</para>
</sect2>
<sect2 id="messages">
<title>Messages</title>
<para>
As mentioned before, messages are represented by subclasses of
<classname>CamelMedium</classname>.
<classname>CamelMedium</classname> itself is a subclass of
<classname>CamelDataWrapper</classname>, a generic class for
connecting a typed data source to a data sink.
<classname>CamelMedium</classname> adds the concept of message
headers versus message body.
(<classname>CamelDataWrapper</classname> has one other
important subclass, <classname>CamelMultipart</classname>,
which is used to provide separate access to the multiple
independent parts of a multipart MIME type.)
<classname>CamelMedium</classname>'s subclasses provide more
specialized handling of various headers:
<classname>CamelMimePart</classname> adds special handling for
the &ldquot;Content-*&rdquot; headers in MIME messages, and
its subclass <classname>CamelMimeMessage</classname> adds
handling for the RFC822 headers.
</para>
<graphic format="gif" fileref="mimemessage"></graphic>
<para>
Consider a message with two parts: a text part (in both plain
text and HTML), and an attached image:
<programlisting>
From: Dan Winship <danw@helixcode.com>
To: Matt Loper <matt@helixcode.com>
Subject: the Camel white paper
MIME-Version: 1.0
Content-Type: multipart/mixed;
boundary="jhTYrnsRrdhDFGa"
This is a multi-part message in MIME format.
--jhTYrnsRrdhDFGa
Content-Type: multipart/alternative;
boundary="sFSenbAFDSgDfg"
--sFSenbAFDSgDfg
Content-Type: text/plain
Hey, Matt
Check out this graphic...
-- Dan
--sFSenbAFDSgDfg
Content-Type: text/html
Hey, Matt<br>
<br>
Check out this graphic...<br>
<br>
-- Dan<br>
<br>
--sFSenbAFDSgDfg--
--jhTYrnsRrdhDFGa
Content-Type: image/png
Content-Transfer-Encoding: base64
F4JLw0ORrkRa8AwAMQJLAaI3UDIGsco9RAaB92...
--jhTYrnsRrdhDFGa--
</programlisting>
<para>
In &Camel;, this would be represented as follows:
</para>
<graphic fileref="samplemsg"></graphic>
</sect2>
<sect2 id="streams">
<title>Streams</title>
<para>
Streams are a generic data transport layer. Two basic stream
classes are <classname>CamelStreamFs</classname>, for
reading and writing files, and
<classname>CamelStreamMem</classname>, for reading from and
writing to objects that are already in memory.
</para>
<para>
Streams can also be chained together. So a CamelMimePart
containing base64-encoded data can filter its output through
a CamelStreamB64. Other parts of the application that want
to read its data will never need to even realize that the
original data was encoded.
</para>
</sect2>
</article>
|