actionmailer-2.3.17-rack-upgrade 2.3.17

Sign up to get free protection for your applications and to get access to all the features.
Files changed (148) hide show
  1. data/CHANGELOG +387 -0
  2. data/MIT-LICENSE +21 -0
  3. data/README +149 -0
  4. data/Rakefile +97 -0
  5. data/install.rb +30 -0
  6. data/lib/action_mailer.rb +62 -0
  7. data/lib/action_mailer/adv_attr_accessor.rb +30 -0
  8. data/lib/action_mailer/base.rb +739 -0
  9. data/lib/action_mailer/helpers.rb +113 -0
  10. data/lib/action_mailer/mail_helper.rb +17 -0
  11. data/lib/action_mailer/part.rb +107 -0
  12. data/lib/action_mailer/part_container.rb +55 -0
  13. data/lib/action_mailer/quoting.rb +62 -0
  14. data/lib/action_mailer/test_case.rb +64 -0
  15. data/lib/action_mailer/test_helper.rb +68 -0
  16. data/lib/action_mailer/utils.rb +7 -0
  17. data/lib/action_mailer/vendor/text-format-0.6.3/text/format.rb +1466 -0
  18. data/lib/action_mailer/vendor/text_format.rb +10 -0
  19. data/lib/action_mailer/vendor/tmail-1.2.7/tmail.rb +6 -0
  20. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/Makefile +18 -0
  21. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/address.rb +392 -0
  22. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/attachments.rb +65 -0
  23. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/base64.rb +46 -0
  24. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/compat.rb +41 -0
  25. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/config.rb +67 -0
  26. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/core_extensions.rb +63 -0
  27. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/encode.rb +590 -0
  28. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/header.rb +962 -0
  29. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/index.rb +9 -0
  30. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/interface.rb +1162 -0
  31. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/loader.rb +3 -0
  32. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/mail.rb +578 -0
  33. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/mailbox.rb +496 -0
  34. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/main.rb +6 -0
  35. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/mbox.rb +3 -0
  36. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/net.rb +250 -0
  37. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/obsolete.rb +132 -0
  38. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/parser.rb +1060 -0
  39. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/parser.y +416 -0
  40. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/port.rb +379 -0
  41. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/quoting.rb +164 -0
  42. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/require_arch.rb +58 -0
  43. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/scanner.rb +49 -0
  44. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/scanner_r.rb +262 -0
  45. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/stringio.rb +280 -0
  46. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/utils.rb +362 -0
  47. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/COPYING +504 -0
  48. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/README +12 -0
  49. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet.rb +67 -0
  50. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/big5freq.rb +927 -0
  51. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/big5prober.rb +42 -0
  52. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/chardistribution.rb +238 -0
  53. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/charsetgroupprober.rb +112 -0
  54. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/charsetprober.rb +75 -0
  55. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/codingstatemachine.rb +64 -0
  56. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/constants.rb +42 -0
  57. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/escprober.rb +89 -0
  58. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/escsm.rb +244 -0
  59. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/eucjpprober.rb +88 -0
  60. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/euckrfreq.rb +596 -0
  61. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/euckrprober.rb +42 -0
  62. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/euctwfreq.rb +430 -0
  63. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/euctwprober.rb +42 -0
  64. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/gb2312freq.rb +474 -0
  65. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/gb2312prober.rb +42 -0
  66. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/hebrewprober.rb +289 -0
  67. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/jisfreq.rb +570 -0
  68. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/jpcntx.rb +229 -0
  69. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/langbulgarianmodel.rb +229 -0
  70. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/langcyrillicmodel.rb +330 -0
  71. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/langgreekmodel.rb +227 -0
  72. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/langhebrewmodel.rb +202 -0
  73. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/langhungarianmodel.rb +226 -0
  74. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/langthaimodel.rb +201 -0
  75. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/latin1prober.rb +147 -0
  76. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/mbcharsetprober.rb +89 -0
  77. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/mbcsgroupprober.rb +45 -0
  78. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/mbcssm.rb +542 -0
  79. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/sbcharsetprober.rb +124 -0
  80. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/sbcsgroupprober.rb +56 -0
  81. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/sjisprober.rb +88 -0
  82. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/universaldetector.rb +168 -0
  83. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/vendor/rchardet-1.3/lib/rchardet/utf8prober.rb +87 -0
  84. data/lib/action_mailer/vendor/tmail-1.2.7/tmail/version.rb +39 -0
  85. data/lib/action_mailer/vendor/tmail.rb +17 -0
  86. data/lib/action_mailer/version.rb +9 -0
  87. data/lib/actionmailer.rb +2 -0
  88. data/test/abstract_unit.rb +62 -0
  89. data/test/asset_host_test.rb +54 -0
  90. data/test/delivery_method_test.rb +51 -0
  91. data/test/fixtures/asset_host_mailer/email_with_asset.html.erb +1 -0
  92. data/test/fixtures/auto_layout_mailer/hello.html.erb +1 -0
  93. data/test/fixtures/auto_layout_mailer/multipart.text.html.erb +1 -0
  94. data/test/fixtures/auto_layout_mailer/multipart.text.plain.erb +1 -0
  95. data/test/fixtures/explicit_layout_mailer/logout.html.erb +1 -0
  96. data/test/fixtures/explicit_layout_mailer/signup.html.erb +1 -0
  97. data/test/fixtures/first_mailer/share.erb +1 -0
  98. data/test/fixtures/helper_mailer/use_example_helper.erb +1 -0
  99. data/test/fixtures/helper_mailer/use_helper.erb +1 -0
  100. data/test/fixtures/helper_mailer/use_helper_method.erb +1 -0
  101. data/test/fixtures/helper_mailer/use_mail_helper.erb +5 -0
  102. data/test/fixtures/helpers/example_helper.rb +5 -0
  103. data/test/fixtures/layouts/auto_layout_mailer.html.erb +1 -0
  104. data/test/fixtures/layouts/auto_layout_mailer.text.erb +1 -0
  105. data/test/fixtures/layouts/spam.html.erb +1 -0
  106. data/test/fixtures/path.with.dots/funky_path_mailer/multipart_with_template_path_with_dots.erb +1 -0
  107. data/test/fixtures/raw_email +14 -0
  108. data/test/fixtures/raw_email10 +20 -0
  109. data/test/fixtures/raw_email12 +32 -0
  110. data/test/fixtures/raw_email13 +29 -0
  111. data/test/fixtures/raw_email2 +114 -0
  112. data/test/fixtures/raw_email3 +70 -0
  113. data/test/fixtures/raw_email4 +59 -0
  114. data/test/fixtures/raw_email5 +19 -0
  115. data/test/fixtures/raw_email6 +20 -0
  116. data/test/fixtures/raw_email7 +66 -0
  117. data/test/fixtures/raw_email8 +47 -0
  118. data/test/fixtures/raw_email9 +28 -0
  119. data/test/fixtures/raw_email_quoted_with_0d0a +14 -0
  120. data/test/fixtures/raw_email_with_invalid_characters_in_content_type +104 -0
  121. data/test/fixtures/raw_email_with_nested_attachment +100 -0
  122. data/test/fixtures/raw_email_with_partially_quoted_subject +14 -0
  123. data/test/fixtures/second_mailer/share.erb +1 -0
  124. data/test/fixtures/templates/signed_up.erb +3 -0
  125. data/test/fixtures/test_mailer/_subtemplate.text.plain.erb +1 -0
  126. data/test/fixtures/test_mailer/body_ivar.erb +2 -0
  127. data/test/fixtures/test_mailer/custom_templating_extension.text.html.haml +6 -0
  128. data/test/fixtures/test_mailer/custom_templating_extension.text.plain.haml +6 -0
  129. data/test/fixtures/test_mailer/implicitly_multipart_example.ignored.erb +1 -0
  130. data/test/fixtures/test_mailer/implicitly_multipart_example.rhtml.bak +1 -0
  131. data/test/fixtures/test_mailer/implicitly_multipart_example.text.html.erb +10 -0
  132. data/test/fixtures/test_mailer/implicitly_multipart_example.text.html.erb~ +10 -0
  133. data/test/fixtures/test_mailer/implicitly_multipart_example.text.plain.erb +2 -0
  134. data/test/fixtures/test_mailer/implicitly_multipart_example.text.yaml.erb +1 -0
  135. data/test/fixtures/test_mailer/included_subtemplate.text.plain.erb +1 -0
  136. data/test/fixtures/test_mailer/rxml_template.builder +2 -0
  137. data/test/fixtures/test_mailer/rxml_template.rxml +2 -0
  138. data/test/fixtures/test_mailer/signed_up.html.erb +3 -0
  139. data/test/fixtures/test_mailer/signed_up_with_url.erb +5 -0
  140. data/test/mail_helper_test.rb +95 -0
  141. data/test/mail_layout_test.rb +123 -0
  142. data/test/mail_render_test.rb +116 -0
  143. data/test/mail_service_test.rb +1145 -0
  144. data/test/quoting_test.rb +105 -0
  145. data/test/test_helper_test.rb +129 -0
  146. data/test/tmail_test.rb +22 -0
  147. data/test/url_test.rb +76 -0
  148. metadata +209 -0
@@ -0,0 +1,280 @@
1
+ # encoding: utf-8
2
+ =begin rdoc
3
+
4
+ = String handling class
5
+
6
+ =end
7
+ #--
8
+ # Copyright (c) 1998-2003 Minero Aoki <aamine@loveruby.net>
9
+ #
10
+ # Permission is hereby granted, free of charge, to any person obtaining
11
+ # a copy of this software and associated documentation files (the
12
+ # "Software"), to deal in the Software without restriction, including
13
+ # without limitation the rights to use, copy, modify, merge, publish,
14
+ # distribute, sublicense, and/or sell copies of the Software, and to
15
+ # permit persons to whom the Software is furnished to do so, subject to
16
+ # the following conditions:
17
+ #
18
+ # The above copyright notice and this permission notice shall be
19
+ # included in all copies or substantial portions of the Software.
20
+ #
21
+ # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
22
+ # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
23
+ # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24
+ # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25
+ # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26
+ # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27
+ # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
28
+ #
29
+ # Note: Originally licensed under LGPL v2+. Using MIT license for Rails
30
+ # with permission of Minero Aoki.
31
+ #++
32
+
33
+ class StringInput#:nodoc:
34
+
35
+ include Enumerable
36
+
37
+ class << self
38
+
39
+ def new( str )
40
+ if block_given?
41
+ begin
42
+ f = super
43
+ yield f
44
+ ensure
45
+ f.close if f
46
+ end
47
+ else
48
+ super
49
+ end
50
+ end
51
+
52
+ alias open new
53
+
54
+ end
55
+
56
+ def initialize( str )
57
+ @src = str
58
+ @pos = 0
59
+ @closed = false
60
+ @lineno = 0
61
+ end
62
+
63
+ attr_reader :lineno
64
+
65
+ def string
66
+ @src
67
+ end
68
+
69
+ def inspect
70
+ "#<#{self.class}:#{@closed ? 'closed' : 'open'},src=#{@src[0,30].inspect}>"
71
+ end
72
+
73
+ def close
74
+ stream_check!
75
+ @pos = nil
76
+ @closed = true
77
+ end
78
+
79
+ def closed?
80
+ @closed
81
+ end
82
+
83
+ def pos
84
+ stream_check!
85
+ [@pos, @src.size].min
86
+ end
87
+
88
+ alias tell pos
89
+
90
+ def seek( offset, whence = IO::SEEK_SET )
91
+ stream_check!
92
+ case whence
93
+ when IO::SEEK_SET
94
+ @pos = offset
95
+ when IO::SEEK_CUR
96
+ @pos += offset
97
+ when IO::SEEK_END
98
+ @pos = @src.size - offset
99
+ else
100
+ raise ArgumentError, "unknown seek flag: #{whence}"
101
+ end
102
+ @pos = 0 if @pos < 0
103
+ @pos = [@pos, @src.size + 1].min
104
+ offset
105
+ end
106
+
107
+ def rewind
108
+ stream_check!
109
+ @pos = 0
110
+ end
111
+
112
+ def eof?
113
+ stream_check!
114
+ @pos > @src.size
115
+ end
116
+
117
+ def each( &block )
118
+ stream_check!
119
+ begin
120
+ @src.each(&block)
121
+ ensure
122
+ @pos = 0
123
+ end
124
+ end
125
+
126
+ def gets
127
+ stream_check!
128
+ if idx = @src.index(?\n, @pos)
129
+ idx += 1 # "\n".size
130
+ line = @src[ @pos ... idx ]
131
+ @pos = idx
132
+ @pos += 1 if @pos == @src.size
133
+ else
134
+ line = @src[ @pos .. -1 ]
135
+ @pos = @src.size + 1
136
+ end
137
+ @lineno += 1
138
+
139
+ line
140
+ end
141
+
142
+ def getc
143
+ stream_check!
144
+ ch = @src[@pos]
145
+ @pos += 1
146
+ @pos += 1 if @pos == @src.size
147
+ ch
148
+ end
149
+
150
+ def read( len = nil )
151
+ stream_check!
152
+ return read_all unless len
153
+ str = @src[@pos, len]
154
+ @pos += len
155
+ @pos += 1 if @pos == @src.size
156
+ str
157
+ end
158
+
159
+ alias sysread read
160
+
161
+ def read_all
162
+ stream_check!
163
+ return nil if eof?
164
+ rest = @src[@pos ... @src.size]
165
+ @pos = @src.size + 1
166
+ rest
167
+ end
168
+
169
+ def stream_check!
170
+ @closed and raise IOError, 'closed stream'
171
+ end
172
+
173
+ end
174
+
175
+
176
+ class StringOutput#:nodoc:
177
+
178
+ class << self
179
+
180
+ def new( str = '' )
181
+ if block_given?
182
+ begin
183
+ f = super
184
+ yield f
185
+ ensure
186
+ f.close if f
187
+ end
188
+ else
189
+ super
190
+ end
191
+ end
192
+
193
+ alias open new
194
+
195
+ end
196
+
197
+ def initialize( str = '' )
198
+ @dest = str
199
+ @closed = false
200
+ end
201
+
202
+ def close
203
+ @closed = true
204
+ end
205
+
206
+ def closed?
207
+ @closed
208
+ end
209
+
210
+ def string
211
+ @dest
212
+ end
213
+
214
+ alias value string
215
+ alias to_str string
216
+
217
+ def size
218
+ @dest.size
219
+ end
220
+
221
+ alias pos size
222
+
223
+ def inspect
224
+ "#<#{self.class}:#{@dest ? 'open' : 'closed'},#{object_id}>"
225
+ end
226
+
227
+ def print( *args )
228
+ stream_check!
229
+ raise ArgumentError, 'wrong # of argument (0 for >1)' if args.empty?
230
+ args.each do |s|
231
+ raise ArgumentError, 'nil not allowed' if s.nil?
232
+ @dest << s.to_s
233
+ end
234
+ nil
235
+ end
236
+
237
+ def puts( *args )
238
+ stream_check!
239
+ args.each do |str|
240
+ @dest << (s = str.to_s)
241
+ @dest << "\n" unless s[-1] == ?\n
242
+ end
243
+ @dest << "\n" if args.empty?
244
+ nil
245
+ end
246
+
247
+ def putc( ch )
248
+ stream_check!
249
+ @dest << ch.chr
250
+ nil
251
+ end
252
+
253
+ def printf( *args )
254
+ stream_check!
255
+ @dest << sprintf(*args)
256
+ nil
257
+ end
258
+
259
+ def write( str )
260
+ stream_check!
261
+ s = str.to_s
262
+ @dest << s
263
+ s.size
264
+ end
265
+
266
+ alias syswrite write
267
+
268
+ def <<( str )
269
+ stream_check!
270
+ @dest << str.to_s
271
+ self
272
+ end
273
+
274
+ private
275
+
276
+ def stream_check!
277
+ @closed and raise IOError, 'closed stream'
278
+ end
279
+
280
+ end
@@ -0,0 +1,362 @@
1
+ # encoding: us-ascii
2
+ #--
3
+ # Copyright (c) 1998-2003 Minero Aoki <aamine@loveruby.net>
4
+ #
5
+ # Permission is hereby granted, free of charge, to any person obtaining
6
+ # a copy of this software and associated documentation files (the
7
+ # "Software"), to deal in the Software without restriction, including
8
+ # without limitation the rights to use, copy, modify, merge, publish,
9
+ # distribute, sublicense, and/or sell copies of the Software, and to
10
+ # permit persons to whom the Software is furnished to do so, subject to
11
+ # the following conditions:
12
+ #
13
+ # The above copyright notice and this permission notice shall be
14
+ # included in all copies or substantial portions of the Software.
15
+ #
16
+ # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23
+ #
24
+ # Note: Originally licensed under LGPL v2+. Using MIT license for Rails
25
+ # with permission of Minero Aoki.
26
+ #++
27
+
28
+ # = TMail - The EMail Swiss Army Knife for Ruby
29
+ #
30
+ # The TMail library provides you with a very complete way to handle and manipulate EMails
31
+ # from within your Ruby programs.
32
+ #
33
+ # Used as the backbone for email handling by the Ruby on Rails and Nitro web frameworks as
34
+ # well as a bunch of other Ruby apps including the Ruby-Talk mailing list to newsgroup email
35
+ # gateway, it is a proven and reliable email handler that won't let you down.
36
+ #
37
+ # Originally created by Minero Aoki, TMail has been recently picked up by Mikel Lindsaar and
38
+ # is being actively maintained. Numerous backlogged bug fixes have been applied as well as
39
+ # Ruby 1.9 compatibility and a swath of documentation to boot.
40
+ #
41
+ # TMail allows you to treat an email totally as an object and allow you to get on with your
42
+ # own programming without having to worry about crafting the perfect email address validation
43
+ # parser, or assembling an email from all it's component parts.
44
+ #
45
+ # TMail handles the most complex part of the email - the header. It generates and parses
46
+ # headers and provides you with instant access to their innards through simple and logically
47
+ # named accessor and setter methods.
48
+ #
49
+ # TMail also provides a wrapper to Net/SMTP as well as Unix Mailbox handling methods to
50
+ # directly read emails from your unix mailbox, parse them and use them.
51
+ #
52
+ # Following is the comprehensive list of methods to access TMail::Mail objects. You can also
53
+ # check out TMail::Mail, TMail::Address and TMail::Headers for other lists.
54
+ module TMail
55
+
56
+ # Provides an exception to throw on errors in Syntax within TMail's parsers
57
+ class SyntaxError < StandardError; end
58
+
59
+ # Provides a new email boundary to separate parts of the email. This is a random
60
+ # string based off the current time, so should be fairly unique.
61
+ #
62
+ # For Example:
63
+ #
64
+ # TMail.new_boundary
65
+ # #=> "mimepart_47bf656968207_25a8fbb80114"
66
+ # TMail.new_boundary
67
+ # #=> "mimepart_47bf66051de4_25a8fbb80240"
68
+ def TMail.new_boundary
69
+ 'mimepart_' + random_tag
70
+ end
71
+
72
+ # Provides a new email message ID. You can use this to generate unique email message
73
+ # id's for your email so you can track them.
74
+ #
75
+ # Optionally takes a fully qualified domain name (default to the current hostname
76
+ # returned by Socket.gethostname) that will be appended to the message ID.
77
+ #
78
+ # For Example:
79
+ #
80
+ # email.message_id = TMail.new_message_id
81
+ # #=> "<47bf66845380e_25a8fbb80332@baci.local.tmail>"
82
+ # email.to_s
83
+ # #=> "Message-Id: <47bf668b633f1_25a8fbb80475@baci.local.tmail>\n\n"
84
+ # email.message_id = TMail.new_message_id("lindsaar.net")
85
+ # #=> "<47bf668b633f1_25a8fbb80475@lindsaar.net.tmail>"
86
+ # email.to_s
87
+ # #=> "Message-Id: <47bf668b633f1_25a8fbb80475@lindsaar.net.tmail>\n\n"
88
+ def TMail.new_message_id( fqdn = nil )
89
+ fqdn ||= ::Socket.gethostname
90
+ "<#{random_tag()}@#{fqdn}.tmail>"
91
+ end
92
+
93
+ #:stopdoc:
94
+ def TMail.random_tag #:nodoc:
95
+ @uniq += 1
96
+ t = Time.now
97
+ sprintf('%x%x_%x%x%d%x',
98
+ t.to_i, t.tv_usec,
99
+ $$, Thread.current.object_id, @uniq, rand(255))
100
+ end
101
+ private_class_method :random_tag
102
+
103
+ @uniq = 0
104
+
105
+ #:startdoc:
106
+
107
+ # Text Utils provides a namespace to define TOKENs, ATOMs, PHRASEs and CONTROL characters that
108
+ # are OK per RFC 2822.
109
+ #
110
+ # It also provides methods you can call to determine if a string is safe
111
+ module TextUtils
112
+
113
+ aspecial = %Q|()<>[]:;.\\,"|
114
+ tspecial = %Q|()<>[];:\\,"/?=|
115
+ lwsp = %Q| \t\r\n|
116
+ control = %Q|\x00-\x1f\x7f-\xff|
117
+
118
+ CONTROL_CHAR = /[#{control}]/n
119
+ ATOM_UNSAFE = /[#{Regexp.quote aspecial}#{control}#{lwsp}]/n
120
+ PHRASE_UNSAFE = /[#{Regexp.quote aspecial}#{control}]/n
121
+ TOKEN_UNSAFE = /[#{Regexp.quote tspecial}#{control}#{lwsp}]/n
122
+
123
+ # Returns true if the string supplied is free from characters not allowed as an ATOM
124
+ def atom_safe?( str )
125
+ not ATOM_UNSAFE === str
126
+ end
127
+
128
+ # If the string supplied has ATOM unsafe characters in it, will return the string quoted
129
+ # in double quotes, otherwise returns the string unmodified
130
+ def quote_atom( str )
131
+ (ATOM_UNSAFE === str) ? dquote(str) : str
132
+ end
133
+
134
+ # If the string supplied has PHRASE unsafe characters in it, will return the string quoted
135
+ # in double quotes, otherwise returns the string unmodified
136
+ def quote_phrase( str )
137
+ (PHRASE_UNSAFE === str) ? dquote(str) : str
138
+ end
139
+
140
+ # Returns true if the string supplied is free from characters not allowed as a TOKEN
141
+ def token_safe?( str )
142
+ not TOKEN_UNSAFE === str
143
+ end
144
+
145
+ # If the string supplied has TOKEN unsafe characters in it, will return the string quoted
146
+ # in double quotes, otherwise returns the string unmodified
147
+ def quote_token( str )
148
+ (TOKEN_UNSAFE === str) ? dquote(str) : str
149
+ end
150
+
151
+ # Wraps supplied string in double quotes unless it is already wrapped
152
+ # Returns double quoted string
153
+ def dquote( str ) #:nodoc:
154
+ unless str =~ /^".*?"$/
155
+ '"' + str.gsub(/["\\]/n) {|s| '\\' + s } + '"'
156
+ else
157
+ str
158
+ end
159
+ end
160
+ private :dquote
161
+
162
+ # Unwraps supplied string from inside double quotes
163
+ # Returns unquoted string
164
+ def unquote( str )
165
+ str =~ /^"(.*?)"$/m ? $1 : str
166
+ end
167
+
168
+ # Provides a method to join a domain name by it's parts and also makes it
169
+ # ATOM safe by quoting it as needed
170
+ def join_domain( arr )
171
+ arr.map {|i|
172
+ if /\A\[.*\]\z/ === i
173
+ i
174
+ else
175
+ quote_atom(i)
176
+ end
177
+ }.join('.')
178
+ end
179
+
180
+ #:stopdoc:
181
+ ZONESTR_TABLE = {
182
+ 'jst' => 9 * 60,
183
+ 'eet' => 2 * 60,
184
+ 'bst' => 1 * 60,
185
+ 'met' => 1 * 60,
186
+ 'gmt' => 0,
187
+ 'utc' => 0,
188
+ 'ut' => 0,
189
+ 'nst' => -(3 * 60 + 30),
190
+ 'ast' => -4 * 60,
191
+ 'edt' => -4 * 60,
192
+ 'est' => -5 * 60,
193
+ 'cdt' => -5 * 60,
194
+ 'cst' => -6 * 60,
195
+ 'mdt' => -6 * 60,
196
+ 'mst' => -7 * 60,
197
+ 'pdt' => -7 * 60,
198
+ 'pst' => -8 * 60,
199
+ 'a' => -1 * 60,
200
+ 'b' => -2 * 60,
201
+ 'c' => -3 * 60,
202
+ 'd' => -4 * 60,
203
+ 'e' => -5 * 60,
204
+ 'f' => -6 * 60,
205
+ 'g' => -7 * 60,
206
+ 'h' => -8 * 60,
207
+ 'i' => -9 * 60,
208
+ # j not use
209
+ 'k' => -10 * 60,
210
+ 'l' => -11 * 60,
211
+ 'm' => -12 * 60,
212
+ 'n' => 1 * 60,
213
+ 'o' => 2 * 60,
214
+ 'p' => 3 * 60,
215
+ 'q' => 4 * 60,
216
+ 'r' => 5 * 60,
217
+ 's' => 6 * 60,
218
+ 't' => 7 * 60,
219
+ 'u' => 8 * 60,
220
+ 'v' => 9 * 60,
221
+ 'w' => 10 * 60,
222
+ 'x' => 11 * 60,
223
+ 'y' => 12 * 60,
224
+ 'z' => 0 * 60
225
+ }
226
+ #:startdoc:
227
+
228
+ # Takes a time zone string from an EMail and converts it to Unix Time (seconds)
229
+ def timezone_string_to_unixtime( str )
230
+ if m = /([\+\-])(\d\d?)(\d\d)/.match(str)
231
+ sec = (m[2].to_i * 60 + m[3].to_i) * 60
232
+ m[1] == '-' ? -sec : sec
233
+ else
234
+ min = ZONESTR_TABLE[str.downcase] or
235
+ raise SyntaxError, "wrong timezone format '#{str}'"
236
+ min * 60
237
+ end
238
+ end
239
+
240
+ #:stopdoc:
241
+ WDAY = %w( Sun Mon Tue Wed Thu Fri Sat TMailBUG )
242
+ MONTH = %w( TMailBUG Jan Feb Mar Apr May Jun
243
+ Jul Aug Sep Oct Nov Dec TMailBUG )
244
+
245
+ def time2str( tm )
246
+ # [ruby-list:7928]
247
+ gmt = Time.at(tm.to_i)
248
+ gmt.gmtime
249
+ offset = tm.to_i - Time.local(*gmt.to_a[0,6].reverse).to_i
250
+
251
+ # DO NOT USE strftime: setlocale() breaks it
252
+ sprintf '%s, %s %s %d %02d:%02d:%02d %+.2d%.2d',
253
+ WDAY[tm.wday], tm.mday, MONTH[tm.month],
254
+ tm.year, tm.hour, tm.min, tm.sec,
255
+ *(offset / 60).divmod(60)
256
+ end
257
+
258
+
259
+ MESSAGE_ID = /<[^\@>]+\@[^>]+>/
260
+
261
+ def message_id?( str )
262
+ MESSAGE_ID === str
263
+ end
264
+
265
+
266
+ MIME_ENCODED = /=\?[^\s?=]+\?[QB]\?[^\s?=]+\?=/i
267
+
268
+ def mime_encoded?( str )
269
+ MIME_ENCODED === str
270
+ end
271
+
272
+
273
+ def decode_params( hash )
274
+ new = Hash.new
275
+ encoded = nil
276
+ hash.each do |key, value|
277
+ if m = /\*(?:(\d+)\*)?\z/.match(key)
278
+ ((encoded ||= {})[m.pre_match] ||= [])[(m[1] || 0).to_i] = value
279
+ else
280
+ new[key] = to_kcode(value)
281
+ end
282
+ end
283
+ if encoded
284
+ encoded.each do |key, strings|
285
+ new[key] = decode_RFC2231(strings.join(''))
286
+ end
287
+ end
288
+
289
+ new
290
+ end
291
+
292
+ NKF_FLAGS = {
293
+ 'EUC' => '-e -m',
294
+ 'SJIS' => '-s -m'
295
+ }
296
+
297
+ def to_kcode( str )
298
+ flag = NKF_FLAGS[TMail.KCODE] or return str
299
+ NKF.nkf(flag, str)
300
+ end
301
+
302
+ RFC2231_ENCODED = /\A(?:iso-2022-jp|euc-jp|shift_jis|us-ascii)?'[a-z]*'/in
303
+
304
+ def decode_RFC2231( str )
305
+ m = RFC2231_ENCODED.match(str) or return str
306
+ begin
307
+ to_kcode(m.post_match.gsub(/%[\da-f]{2}/in) {|s| s[1,2].hex.chr })
308
+ rescue
309
+ m.post_match.gsub(/%[\da-f]{2}/in, "")
310
+ end
311
+ end
312
+
313
+ def quote_boundary
314
+ # Make sure the Content-Type boundary= parameter is quoted if it contains illegal characters
315
+ # (to ensure any special characters in the boundary text are escaped from the parser
316
+ # (such as = in MS Outlook's boundary text))
317
+ if @body =~ /^(.*)boundary=(.*)$/m
318
+ preamble = $1
319
+ remainder = $2
320
+ if remainder =~ /;/
321
+ remainder =~ /^(.*?)(;.*)$/m
322
+ boundary_text = $1
323
+ post = $2.chomp
324
+ else
325
+ boundary_text = remainder.chomp
326
+ end
327
+ if boundary_text =~ /[\/\?\=]/
328
+ boundary_text = "\"#{boundary_text}\"" unless boundary_text =~ /^".*?"$/
329
+ @body = "#{preamble}boundary=#{boundary_text}#{post}"
330
+ end
331
+ end
332
+ end
333
+
334
+ # AppleMail generates illegal character contained Content-Type parameter like:
335
+ # name==?ISO-2022-JP?B?...=?=
336
+ # so quote. (This case is only value fits in one line.)
337
+ def quote_unquoted_bencode
338
+ @body = @body.gsub(%r"(;\s+[-a-z]+=)(=\?.+?)([;\r\n ]|\z)"m) {
339
+ head, should_quoted, tail = $~.captures
340
+ # head: "; name="
341
+ # should_quoted: "=?ISO-2022-JP?B?...=?="
342
+
343
+ head << quote_token(should_quoted) << tail
344
+ }
345
+ end
346
+
347
+ # AppleMail generates name=filename attributes in the content type that
348
+ # contain spaces. Need to handle this so the TMail Parser can.
349
+ def quote_unquoted_name
350
+ @body = @body.gsub(%r|(name=)([\w\s.]+)(.*)|m) {
351
+ head, should_quoted, tail = $~.captures
352
+ # head: "; name="
353
+ # should_quoted: "=?ISO-2022-JP?B?...=?="
354
+ head << quote_token(should_quoted) << tail
355
+ }
356
+ end
357
+
358
+ #:startdoc:
359
+
360
+ end
361
+
362
+ end