net-imap 0.4.12 → 0.5.6
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/Gemfile +8 -1
- data/README.md +10 -4
- data/docs/styles.css +75 -14
- data/lib/net/imap/authenticators.rb +2 -2
- data/lib/net/imap/command_data.rb +61 -48
- data/lib/net/imap/config/attr_accessors.rb +75 -0
- data/lib/net/imap/config/attr_inheritance.rb +90 -0
- data/lib/net/imap/config/attr_type_coercion.rb +61 -0
- data/lib/net/imap/config.rb +470 -0
- data/lib/net/imap/data_encoding.rb +3 -3
- data/lib/net/imap/data_lite.rb +226 -0
- data/lib/net/imap/deprecated_client_options.rb +8 -5
- data/lib/net/imap/errors.rb +6 -0
- data/lib/net/imap/esearch_result.rb +180 -0
- data/lib/net/imap/fetch_data.rb +126 -47
- data/lib/net/imap/response_data.rb +124 -237
- data/lib/net/imap/response_parser/parser_utils.rb +11 -6
- data/lib/net/imap/response_parser.rb +187 -34
- data/lib/net/imap/sasl/anonymous_authenticator.rb +3 -3
- data/lib/net/imap/sasl/authentication_exchange.rb +52 -20
- data/lib/net/imap/sasl/authenticators.rb +8 -4
- data/lib/net/imap/sasl/client_adapter.rb +77 -26
- data/lib/net/imap/sasl/cram_md5_authenticator.rb +4 -4
- data/lib/net/imap/sasl/digest_md5_authenticator.rb +218 -56
- data/lib/net/imap/sasl/external_authenticator.rb +2 -2
- data/lib/net/imap/sasl/gs2_header.rb +7 -7
- data/lib/net/imap/sasl/login_authenticator.rb +4 -3
- data/lib/net/imap/sasl/oauthbearer_authenticator.rb +6 -6
- data/lib/net/imap/sasl/plain_authenticator.rb +7 -7
- data/lib/net/imap/sasl/protocol_adapters.rb +60 -4
- data/lib/net/imap/sasl/scram_authenticator.rb +8 -8
- data/lib/net/imap/sasl.rb +7 -4
- data/lib/net/imap/sasl_adapter.rb +0 -1
- data/lib/net/imap/search_result.rb +2 -2
- data/lib/net/imap/sequence_set.rb +221 -82
- data/lib/net/imap/stringprep/nameprep.rb +1 -1
- data/lib/net/imap/stringprep/trace.rb +4 -4
- data/lib/net/imap/uidplus_data.rb +244 -0
- data/lib/net/imap/vanished_data.rb +56 -0
- data/lib/net/imap.rb +1010 -320
- data/net-imap.gemspec +3 -3
- data/rakelib/rfcs.rake +2 -0
- data/rakelib/string_prep_tables_generator.rb +2 -0
- metadata +12 -10
- data/.github/dependabot.yml +0 -6
- data/.github/workflows/pages.yml +0 -46
- data/.github/workflows/push_gem.yml +0 -48
- data/.github/workflows/test.yml +0 -31
- data/.gitignore +0 -12
- data/.mailmap +0 -13
@@ -0,0 +1,226 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Some of the code in this file was copied from the polyfill-data gem.
|
4
|
+
#
|
5
|
+
# MIT License
|
6
|
+
#
|
7
|
+
# Copyright (c) 2023 Jim Gay, Joel Drapper, Nicholas Evans
|
8
|
+
#
|
9
|
+
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
10
|
+
# of this software and associated documentation files (the "Software"), to deal
|
11
|
+
# in the Software without restriction, including without limitation the rights
|
12
|
+
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
13
|
+
# copies of the Software, and to permit persons to whom the Software is
|
14
|
+
# furnished to do so, subject to the following conditions:
|
15
|
+
#
|
16
|
+
# The above copyright notice and this permission notice shall be included in all
|
17
|
+
# copies or substantial portions of the Software.
|
18
|
+
#
|
19
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
20
|
+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
21
|
+
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
22
|
+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
23
|
+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
24
|
+
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
25
|
+
# SOFTWARE.
|
26
|
+
|
27
|
+
|
28
|
+
module Net
|
29
|
+
class IMAP
|
30
|
+
data_or_object = RUBY_VERSION >= "3.2.0" ? ::Data : Object
|
31
|
+
class DataLite < data_or_object
|
32
|
+
def encode_with(coder) coder.map = to_h.transform_keys(&:to_s) end
|
33
|
+
def init_with(coder) initialize(**coder.map.transform_keys(&:to_sym)) end
|
34
|
+
end
|
35
|
+
|
36
|
+
Data = DataLite
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
# :nocov:
|
41
|
+
# Need to skip test coverage for the rest, because it isn't loaded by ruby 3.2+.
|
42
|
+
return if RUBY_VERSION >= "3.2.0"
|
43
|
+
|
44
|
+
module Net
|
45
|
+
class IMAP
|
46
|
+
# DataLite is a temporary substitute for ruby 3.2's +Data+ class. DataLite
|
47
|
+
# is aliased as Net::IMAP::Data, so that code using it won't need to be
|
48
|
+
# updated when it is removed.
|
49
|
+
#
|
50
|
+
# See {ruby 3.2's documentation for Data}[https://docs.ruby-lang.org/en/3.2/Data.html].
|
51
|
+
#
|
52
|
+
# [When running ruby 3.1]
|
53
|
+
# This class reimplements the API for ruby 3.2's +Data+, and should be
|
54
|
+
# compatible for nearly all use-cases. This reimplementation <em>will be
|
55
|
+
# removed</em> in +net-imap+ 0.6, when support for ruby 3.1 is dropped.
|
56
|
+
#
|
57
|
+
# _NOTE:_ +net-imap+ no longer supports ruby versions prior to 3.1.
|
58
|
+
# [When running ruby >= 3.2]
|
59
|
+
# This class inherits from +Data+ and _only_ defines the methods needed
|
60
|
+
# for YAML serialization. This will be dropped when +psych+ adds support
|
61
|
+
# for +Data+.
|
62
|
+
#
|
63
|
+
# Some of the code in this class was copied or adapted from the
|
64
|
+
# {polyfill-data gem}[https://rubygems.org/gems/polyfill-data], by Jim Gay
|
65
|
+
# and Joel Drapper, under the MIT license terms.
|
66
|
+
class DataLite
|
67
|
+
singleton_class.undef_method :new
|
68
|
+
|
69
|
+
TYPE_ERROR = "%p is not a symbol nor a string"
|
70
|
+
ATTRSET_ERROR = "invalid data member: %p"
|
71
|
+
DUP_ERROR = "duplicate member: %p"
|
72
|
+
ARITY_ERROR = "wrong number of arguments (given %d, expected %s)"
|
73
|
+
private_constant :TYPE_ERROR, :ATTRSET_ERROR, :DUP_ERROR, :ARITY_ERROR
|
74
|
+
|
75
|
+
# Defines a new Data class.
|
76
|
+
#
|
77
|
+
# _NOTE:_ Unlike ruby 3.2's +Data.define+, DataLite.define only supports
|
78
|
+
# member names which are valid local variable names. Member names can't
|
79
|
+
# be keywords (e.g: +next+ or +class+) or start with capital letters, "@",
|
80
|
+
# etc.
|
81
|
+
def self.define(*args, &block)
|
82
|
+
members = args.each_with_object({}) do |arg, members|
|
83
|
+
arg = arg.to_str unless arg in Symbol | String if arg.respond_to?(:to_str)
|
84
|
+
arg = arg.to_sym if arg in String
|
85
|
+
arg in Symbol or raise TypeError, TYPE_ERROR % [arg]
|
86
|
+
arg in %r{=} and raise ArgumentError, ATTRSET_ERROR % [arg]
|
87
|
+
members.key?(arg) and raise ArgumentError, DUP_ERROR % [arg]
|
88
|
+
members[arg] = true
|
89
|
+
end
|
90
|
+
members = members.keys.freeze
|
91
|
+
|
92
|
+
klass = ::Class.new(self)
|
93
|
+
|
94
|
+
klass.singleton_class.undef_method :define
|
95
|
+
klass.define_singleton_method(:members) { members }
|
96
|
+
|
97
|
+
def klass.new(*args, **kwargs, &block)
|
98
|
+
if kwargs.size.positive?
|
99
|
+
if args.size.positive?
|
100
|
+
raise ArgumentError, ARITY_ERROR % [args.size, 0]
|
101
|
+
end
|
102
|
+
elsif members.size < args.size
|
103
|
+
expected = members.size.zero? ? 0 : 0..members.size
|
104
|
+
raise ArgumentError, ARITY_ERROR % [args.size, expected]
|
105
|
+
else
|
106
|
+
kwargs = Hash[members.take(args.size).zip(args)]
|
107
|
+
end
|
108
|
+
allocate.tap do |instance|
|
109
|
+
instance.__send__(:initialize, **kwargs, &block)
|
110
|
+
end.freeze
|
111
|
+
end
|
112
|
+
|
113
|
+
klass.singleton_class.alias_method :[], :new
|
114
|
+
klass.attr_reader(*members)
|
115
|
+
|
116
|
+
# Dynamically defined initializer methods are in an included module,
|
117
|
+
# rather than directly on DataLite (like in ruby 3.2+):
|
118
|
+
# * simpler to handle required kwarg ArgumentErrors
|
119
|
+
# * easier to ensure consistent ivar assignment order (object shape)
|
120
|
+
# * faster than instance_variable_set
|
121
|
+
klass.include(Module.new do
|
122
|
+
if members.any?
|
123
|
+
kwargs = members.map{"#{_1.name}:"}.join(", ")
|
124
|
+
params = members.map(&:name).join(", ")
|
125
|
+
ivars = members.map{"@#{_1.name}"}.join(", ")
|
126
|
+
attrs = members.map{"attrs[:#{_1.name}]"}.join(", ")
|
127
|
+
module_eval <<~RUBY, __FILE__, __LINE__ + 1
|
128
|
+
protected
|
129
|
+
def initialize(#{kwargs}) #{ivars} = #{params}; freeze end
|
130
|
+
def marshal_load(attrs) #{ivars} = #{attrs}; freeze end
|
131
|
+
RUBY
|
132
|
+
end
|
133
|
+
end)
|
134
|
+
|
135
|
+
klass.module_eval do _1.module_eval(&block) end if block_given?
|
136
|
+
|
137
|
+
klass
|
138
|
+
end
|
139
|
+
|
140
|
+
##
|
141
|
+
# singleton-method: new
|
142
|
+
# call-seq:
|
143
|
+
# new(*args) -> instance
|
144
|
+
# new(**kwargs) -> instance
|
145
|
+
#
|
146
|
+
# Constuctor for classes defined with ::define.
|
147
|
+
#
|
148
|
+
# Aliased as ::[].
|
149
|
+
|
150
|
+
##
|
151
|
+
# singleton-method: []
|
152
|
+
# call-seq:
|
153
|
+
# ::[](*args) -> instance
|
154
|
+
# ::[](**kwargs) -> instance
|
155
|
+
#
|
156
|
+
# Constuctor for classes defined with ::define.
|
157
|
+
#
|
158
|
+
# Alias for ::new
|
159
|
+
|
160
|
+
##
|
161
|
+
def members; self.class.members end
|
162
|
+
def to_h(&block) block ? __to_h__.to_h(&block) : __to_h__ end
|
163
|
+
def hash; [self.class, __to_h__].hash end
|
164
|
+
def ==(other) self.class == other.class && to_h == other.to_h end
|
165
|
+
def eql?(other) self.class == other.class && hash == other.hash end
|
166
|
+
def deconstruct; __to_h__.values end
|
167
|
+
|
168
|
+
def deconstruct_keys(keys)
|
169
|
+
raise TypeError unless keys.is_a?(Array) || keys.nil?
|
170
|
+
return __to_h__ if keys&.first.nil?
|
171
|
+
__to_h__.slice(*keys)
|
172
|
+
end
|
173
|
+
|
174
|
+
def with(**kwargs)
|
175
|
+
return self if kwargs.empty?
|
176
|
+
self.class.new(**__to_h__.merge(kwargs))
|
177
|
+
end
|
178
|
+
|
179
|
+
def inspect
|
180
|
+
__inspect_guard__(self) do |seen|
|
181
|
+
return "#<data #{self.class}:...>" if seen
|
182
|
+
attrs = __to_h__.map {|kv| "%s=%p" % kv }.join(", ")
|
183
|
+
display = ["data", self.class.name, attrs].compact.join(" ")
|
184
|
+
"#<#{display}>"
|
185
|
+
end
|
186
|
+
end
|
187
|
+
alias_method :to_s, :inspect
|
188
|
+
|
189
|
+
private
|
190
|
+
|
191
|
+
def initialize_copy(source) super.freeze end
|
192
|
+
def marshal_dump; __to_h__ end
|
193
|
+
|
194
|
+
def __to_h__; Hash[members.map {|m| [m, send(m)] }] end
|
195
|
+
|
196
|
+
# Yields +true+ if +obj+ has been seen already, +false+ if it hasn't.
|
197
|
+
# Marks +obj+ as seen inside the block, so circuler references don't
|
198
|
+
# recursively trigger a SystemStackError (stack level too deep).
|
199
|
+
#
|
200
|
+
# Making circular references inside a Data object _should_ be very
|
201
|
+
# uncommon, but we'll support them for the sake of completeness.
|
202
|
+
def __inspect_guard__(obj)
|
203
|
+
preexisting = Thread.current[:__net_imap_data__inspect__]
|
204
|
+
Thread.current[:__net_imap_data__inspect__] ||= {}.compare_by_identity
|
205
|
+
inspect_guard = Thread.current[:__net_imap_data__inspect__]
|
206
|
+
if inspect_guard.include?(obj)
|
207
|
+
yield true
|
208
|
+
else
|
209
|
+
begin
|
210
|
+
inspect_guard[obj] = true
|
211
|
+
yield false
|
212
|
+
ensure
|
213
|
+
inspect_guard.delete(obj)
|
214
|
+
end
|
215
|
+
end
|
216
|
+
ensure
|
217
|
+
unless preexisting.equal?(inspect_guard)
|
218
|
+
Thread.current[:__net_imap_data__inspect__] = preexisting
|
219
|
+
end
|
220
|
+
end
|
221
|
+
|
222
|
+
end
|
223
|
+
|
224
|
+
end
|
225
|
+
end
|
226
|
+
# :nocov:
|
@@ -16,8 +16,8 @@ module Net
|
|
16
16
|
#
|
17
17
|
# ==== Obsolete arguments
|
18
18
|
#
|
19
|
-
#
|
20
|
-
# deprecated by a future release.
|
19
|
+
# Use of obsolete arguments does not print a warning. Obsolete arguments
|
20
|
+
# will be deprecated by a future release.
|
21
21
|
#
|
22
22
|
# If a second positional argument is given and it is a hash (or is
|
23
23
|
# convertible via +#to_hash+), it is converted to keyword arguments.
|
@@ -83,10 +83,12 @@ module Net
|
|
83
83
|
elsif deprecated.empty?
|
84
84
|
super host, port: port_or_options
|
85
85
|
elsif deprecated.shift
|
86
|
-
warn
|
86
|
+
warn("DEPRECATED: Call Net::IMAP.new with keyword options",
|
87
|
+
uplevel: 1, category: :deprecated)
|
87
88
|
super host, port: port_or_options, ssl: create_ssl_params(*deprecated)
|
88
89
|
else
|
89
|
-
warn
|
90
|
+
warn("DEPRECATED: Call Net::IMAP.new with keyword options",
|
91
|
+
uplevel: 1, category: :deprecated)
|
90
92
|
super host, port: port_or_options, ssl: false
|
91
93
|
end
|
92
94
|
end
|
@@ -113,7 +115,8 @@ module Net
|
|
113
115
|
elsif deprecated.first.respond_to?(:to_hash)
|
114
116
|
super(**Hash.try_convert(deprecated.first))
|
115
117
|
else
|
116
|
-
warn
|
118
|
+
warn("DEPRECATED: Call Net::IMAP#starttls with keyword options",
|
119
|
+
uplevel: 1, category: :deprecated)
|
117
120
|
super(**create_ssl_params(*deprecated))
|
118
121
|
end
|
119
122
|
end
|
data/lib/net/imap/errors.rb
CHANGED
@@ -7,6 +7,12 @@ module Net
|
|
7
7
|
class Error < StandardError
|
8
8
|
end
|
9
9
|
|
10
|
+
class LoginDisabledError < Error
|
11
|
+
def initialize(msg = "Remote server has disabled the LOGIN command", ...)
|
12
|
+
super
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
10
16
|
# Error raised when data is in the incorrect format.
|
11
17
|
class DataFormatError < Error
|
12
18
|
end
|
@@ -0,0 +1,180 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Net
|
4
|
+
class IMAP
|
5
|
+
# An "extended search" response (+ESEARCH+). ESearchResult should be
|
6
|
+
# returned (instead of SearchResult) by IMAP#search, IMAP#uid_search,
|
7
|
+
# IMAP#sort, and IMAP#uid_sort under any of the following conditions:
|
8
|
+
#
|
9
|
+
# * Return options were specified for IMAP#search or IMAP#uid_search.
|
10
|
+
# The server must support a search extension which allows
|
11
|
+
# RFC4466[https://www.rfc-editor.org/rfc/rfc4466.html] +return+ options,
|
12
|
+
# such as +ESEARCH+, +PARTIAL+, or +IMAP4rev2+.
|
13
|
+
# * Return options were specified for IMAP#sort or IMAP#uid_sort.
|
14
|
+
# The server must support the +ESORT+ extension
|
15
|
+
# {[RFC5267]}[https://www.rfc-editor.org/rfc/rfc5267.html#section-3].
|
16
|
+
#
|
17
|
+
# *NOTE:* IMAP#search and IMAP#uid_search do not support +ESORT+ yet.
|
18
|
+
# * The server supports +IMAP4rev2+ but _not_ +IMAP4rev1+, or +IMAP4rev2+
|
19
|
+
# has been enabled. +IMAP4rev2+ requires +ESEARCH+ results.
|
20
|
+
#
|
21
|
+
# Note that some servers may claim to support a search extension which
|
22
|
+
# requires an +ESEARCH+ result, such as +PARTIAL+, but still only return a
|
23
|
+
# +SEARCH+ result when +return+ options are specified.
|
24
|
+
#
|
25
|
+
# Some search extensions may result in the server sending ESearchResult
|
26
|
+
# responses after the initiating command has completed. Use
|
27
|
+
# IMAP#add_response_handler to handle these responses.
|
28
|
+
class ESearchResult < Data.define(:tag, :uid, :data)
|
29
|
+
def initialize(tag: nil, uid: nil, data: nil)
|
30
|
+
tag => String | nil; tag = -tag if tag
|
31
|
+
uid => true | false | nil; uid = !!uid
|
32
|
+
data => Array | nil; data ||= []; data.freeze
|
33
|
+
super
|
34
|
+
end
|
35
|
+
|
36
|
+
# :call-seq: to_a -> Array of integers
|
37
|
+
#
|
38
|
+
# When either #all or #partial contains a SequenceSet of message sequence
|
39
|
+
# numbers or UIDs, +to_a+ returns that set as an array of integers.
|
40
|
+
#
|
41
|
+
# When both #all and #partial are +nil+, either because the server
|
42
|
+
# returned no results or because +ALL+ and +PARTIAL+ were not included in
|
43
|
+
# the IMAP#search +RETURN+ options, #to_a returns an empty array.
|
44
|
+
#
|
45
|
+
# Note that SearchResult also implements +to_a+, so it can be used without
|
46
|
+
# checking if the server returned +SEARCH+ or +ESEARCH+ data.
|
47
|
+
def to_a; all&.numbers || partial&.to_a || [] end
|
48
|
+
|
49
|
+
##
|
50
|
+
# attr_reader: tag
|
51
|
+
#
|
52
|
+
# The tag string for the command that caused this response to be returned.
|
53
|
+
#
|
54
|
+
# When +nil+, this response was not caused by a particular command.
|
55
|
+
|
56
|
+
##
|
57
|
+
# attr_reader: uid
|
58
|
+
#
|
59
|
+
# Indicates whether #data in this response refers to UIDs (when +true+) or
|
60
|
+
# to message sequence numbers (when +false+).
|
61
|
+
|
62
|
+
##
|
63
|
+
alias uid? uid
|
64
|
+
|
65
|
+
##
|
66
|
+
# attr_reader: data
|
67
|
+
#
|
68
|
+
# Search return data, as an array of <tt>[name, value]</tt> pairs. Most
|
69
|
+
# return data corresponds to a search +return+ option with the same name.
|
70
|
+
#
|
71
|
+
# Note that some return data names may be used more than once per result.
|
72
|
+
#
|
73
|
+
# This data can be more simply retrieved by #min, #max, #all, #count,
|
74
|
+
# #modseq, and other methods.
|
75
|
+
|
76
|
+
# :call-seq: min -> integer or nil
|
77
|
+
#
|
78
|
+
# The lowest message number/UID that satisfies the SEARCH criteria.
|
79
|
+
#
|
80
|
+
# Returns +nil+ when the associated search command has no results, or when
|
81
|
+
# the +MIN+ return option wasn't specified.
|
82
|
+
#
|
83
|
+
# Requires +ESEARCH+ {[RFC4731]}[https://www.rfc-editor.org/rfc/rfc4731.html#section-3.1] or
|
84
|
+
# +IMAP4rev2+ {[RFC9051]}[https://www.rfc-editor.org/rfc/rfc9051.html#section-7.3.4].
|
85
|
+
def min; data.assoc("MIN")&.last end
|
86
|
+
|
87
|
+
# :call-seq: max -> integer or nil
|
88
|
+
#
|
89
|
+
# The highest message number/UID that satisfies the SEARCH criteria.
|
90
|
+
#
|
91
|
+
# Returns +nil+ when the associated search command has no results, or when
|
92
|
+
# the +MAX+ return option wasn't specified.
|
93
|
+
#
|
94
|
+
# Requires +ESEARCH+ {[RFC4731]}[https://www.rfc-editor.org/rfc/rfc4731.html#section-3.1] or
|
95
|
+
# +IMAP4rev2+ {[RFC9051]}[https://www.rfc-editor.org/rfc/rfc9051.html#section-7.3.4].
|
96
|
+
def max; data.assoc("MAX")&.last end
|
97
|
+
|
98
|
+
# :call-seq: all -> sequence set or nil
|
99
|
+
#
|
100
|
+
# A SequenceSet containing all message sequence numbers or UIDs that
|
101
|
+
# satisfy the SEARCH criteria.
|
102
|
+
#
|
103
|
+
# Returns +nil+ when the associated search command has no results, or when
|
104
|
+
# the +ALL+ return option was not specified but other return options were.
|
105
|
+
#
|
106
|
+
# Requires +ESEARCH+ {[RFC4731]}[https://www.rfc-editor.org/rfc/rfc4731.html#section-3.1] or
|
107
|
+
# +IMAP4rev2+ {[RFC9051]}[https://www.rfc-editor.org/rfc/rfc9051.html#section-7.3.4].
|
108
|
+
#
|
109
|
+
# See also: #to_a
|
110
|
+
def all; data.assoc("ALL")&.last end
|
111
|
+
|
112
|
+
# :call-seq: count -> integer or nil
|
113
|
+
#
|
114
|
+
# Returns the number of messages that satisfy the SEARCH criteria.
|
115
|
+
#
|
116
|
+
# Returns +nil+ when the associated search command has no results, or when
|
117
|
+
# the +COUNT+ return option wasn't specified.
|
118
|
+
#
|
119
|
+
# Requires +ESEARCH+ {[RFC4731]}[https://www.rfc-editor.org/rfc/rfc4731.html#section-3.1] or
|
120
|
+
# +IMAP4rev2+ {[RFC9051]}[https://www.rfc-editor.org/rfc/rfc9051.html#section-7.3.4].
|
121
|
+
def count; data.assoc("COUNT")&.last end
|
122
|
+
|
123
|
+
# :call-seq: modseq -> integer or nil
|
124
|
+
#
|
125
|
+
# The highest +mod-sequence+ of all messages being returned.
|
126
|
+
#
|
127
|
+
# Returns +nil+ when the associated search command has no results, or when
|
128
|
+
# the +MODSEQ+ search criterion wasn't specified.
|
129
|
+
#
|
130
|
+
# Note that there is no search +return+ option for +MODSEQ+. It will be
|
131
|
+
# returned whenever the +CONDSTORE+ extension has been enabled. Using the
|
132
|
+
# +MODSEQ+ search criteria will implicitly enable +CONDSTORE+.
|
133
|
+
#
|
134
|
+
# Requires +CONDSTORE+ {[RFC7162]}[https://www.rfc-editor.org/rfc/rfc7162.html]
|
135
|
+
# and +ESEARCH+ {[RFC4731]}[https://www.rfc-editor.org/rfc/rfc4731.html#section-3.2].
|
136
|
+
def modseq; data.assoc("MODSEQ")&.last end
|
137
|
+
|
138
|
+
# Returned by ESearchResult#partial.
|
139
|
+
#
|
140
|
+
# Requires +PARTIAL+ {[RFC9394]}[https://www.rfc-editor.org/rfc/rfc9394.html]
|
141
|
+
# or <tt>CONTEXT=SEARCH</tt>/<tt>CONTEXT=SORT</tt>
|
142
|
+
# {[RFC5267]}[https://www.rfc-editor.org/rfc/rfc5267.html]
|
143
|
+
#
|
144
|
+
# See also: #to_a
|
145
|
+
class PartialResult < Data.define(:range, :results)
|
146
|
+
def initialize(range:, results:)
|
147
|
+
range => Range
|
148
|
+
results = SequenceSet[results] unless results.nil?
|
149
|
+
super
|
150
|
+
end
|
151
|
+
|
152
|
+
##
|
153
|
+
# method: range
|
154
|
+
# :call-seq: range -> range
|
155
|
+
|
156
|
+
##
|
157
|
+
# method: results
|
158
|
+
# :call-seq: results -> sequence set or nil
|
159
|
+
|
160
|
+
# Converts #results to an array of integers.
|
161
|
+
#
|
162
|
+
# See also: ESearchResult#to_a.
|
163
|
+
def to_a; results&.numbers || [] end
|
164
|
+
end
|
165
|
+
|
166
|
+
# :call-seq: partial -> PartialResult or nil
|
167
|
+
#
|
168
|
+
# A PartialResult containing a subset of the message sequence numbers or
|
169
|
+
# UIDs that satisfy the SEARCH criteria.
|
170
|
+
#
|
171
|
+
# Requires +PARTIAL+ {[RFC9394]}[https://www.rfc-editor.org/rfc/rfc9394.html]
|
172
|
+
# or <tt>CONTEXT=SEARCH</tt>/<tt>CONTEXT=SORT</tt>
|
173
|
+
# {[RFC5267]}[https://www.rfc-editor.org/rfc/rfc5267.html]
|
174
|
+
#
|
175
|
+
# See also: #to_a
|
176
|
+
def partial; data.assoc("PARTIAL")&.last end
|
177
|
+
|
178
|
+
end
|
179
|
+
end
|
180
|
+
end
|