sisimai 5.6.0-java → 5.7.0-java
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.github/workflows/rake-test.yml +0 -4
- data/ChangeLog.md +32 -0
- data/README-JA.md +19 -16
- data/README.md +23 -21
- data/lib/sisimai/address.rb +8 -31
- data/lib/sisimai/arf.rb +3 -5
- data/lib/sisimai/fact.rb +12 -37
- data/lib/sisimai/lhost/activehunter.rb +0 -2
- data/lib/sisimai/lhost/amazonses.rb +6 -8
- data/lib/sisimai/lhost/apachejames.rb +0 -1
- data/lib/sisimai/lhost/biglobe.rb +0 -16
- data/lib/sisimai/lhost/courier.rb +5 -4
- data/lib/sisimai/lhost/deutschetelekom.rb +120 -0
- data/lib/sisimai/lhost/domino.rb +0 -3
- data/lib/sisimai/lhost/dragonfly.rb +0 -27
- data/lib/sisimai/lhost/einsundeins.rb +1 -10
- data/lib/sisimai/lhost/exchange2003.rb +4 -4
- data/lib/sisimai/lhost/exchange2007.rb +3 -4
- data/lib/sisimai/lhost/exim.rb +30 -78
- data/lib/sisimai/lhost/ezweb.rb +12 -49
- data/lib/sisimai/lhost/fml.rb +4 -29
- data/lib/sisimai/lhost/gmail.rb +0 -23
- data/lib/sisimai/lhost/gmx.rb +7 -24
- data/lib/sisimai/lhost/googlegroups.rb +3 -3
- data/lib/sisimai/lhost/googleworkspace.rb +0 -4
- data/lib/sisimai/lhost/imailserver.rb +3 -9
- data/lib/sisimai/lhost/kddi.rb +6 -20
- data/lib/sisimai/lhost/mailfoundry.rb +0 -2
- data/lib/sisimai/lhost/mailmarshal.rb +1 -3
- data/lib/sisimai/lhost/messagingserver.rb +4 -15
- data/lib/sisimai/lhost/mfilter.rb +0 -1
- data/lib/sisimai/lhost/mimecast.rb +0 -1
- data/lib/sisimai/lhost/notes.rb +1 -2
- data/lib/sisimai/lhost/opensmtpd.rb +0 -40
- data/lib/sisimai/lhost/postfix.rb +10 -11
- data/lib/sisimai/lhost/qmail.rb +14 -81
- data/lib/sisimai/lhost/sendmail.rb +4 -4
- data/lib/sisimai/lhost/trendmicro.rb +3 -3
- data/lib/sisimai/lhost/v5sendmail.rb +0 -1
- data/lib/sisimai/lhost/verizon.rb +1 -2
- data/lib/sisimai/lhost/x1.rb +1 -2
- data/lib/sisimai/lhost/x2.rb +0 -2
- data/lib/sisimai/lhost/x3.rb +4 -9
- data/lib/sisimai/lhost/x6.rb +0 -1
- data/lib/sisimai/lhost/zoho.rb +0 -12
- data/lib/sisimai/lhost.rb +38 -19
- data/lib/sisimai/message.rb +1 -1
- data/lib/sisimai/order.rb +4 -1
- data/lib/sisimai/reason/authfailure.rb +2 -2
- data/lib/sisimai/reason/contenterror.rb +2 -0
- data/lib/sisimai/reason/emailtoolarge.rb +1 -1
- data/lib/sisimai/reason/expired.rb +13 -2
- data/lib/sisimai/reason/hostunknown.rb +9 -0
- data/lib/sisimai/reason/mailboxfull.rb +3 -2
- data/lib/sisimai/reason/networkerror.rb +13 -1
- data/lib/sisimai/reason/norelaying.rb +4 -3
- data/lib/sisimai/reason/notaccept.rb +10 -3
- data/lib/sisimai/reason/notcompliantrfc.rb +1 -0
- data/lib/sisimai/reason/policyviolation.rb +6 -1
- data/lib/sisimai/reason/rejected.rb +6 -0
- data/lib/sisimai/reason/securityerror.rb +1 -0
- data/lib/sisimai/reason/suspend.rb +4 -0
- data/lib/sisimai/reason/syntaxerror.rb +1 -8
- data/lib/sisimai/reason/systemerror.rb +18 -0
- data/lib/sisimai/reason/userunknown.rb +2 -0
- data/lib/sisimai/reason.rb +7 -7
- data/lib/sisimai/rfc1123.rb +1 -1
- data/lib/sisimai/rfc1894.rb +7 -6
- data/lib/sisimai/rfc2045.rb +2 -2
- data/lib/sisimai/rfc3464/thirdparty.rb +1 -1
- data/lib/sisimai/rfc3464.rb +10 -14
- data/lib/sisimai/rfc3834.rb +3 -4
- data/lib/sisimai/rfc791.rb +3 -38
- data/lib/sisimai/rhost/microsoft.rb +4 -0
- data/lib/sisimai/rhost.rb +1 -1
- data/lib/sisimai/smtp/status.rb +23 -19
- data/lib/sisimai/string.rb +0 -12
- data/lib/sisimai/version.rb +1 -1
- data/set-of-emails/maildir/bsd/lhost-deutschetelekom-01.eml +66 -0
- data/set-of-emails/maildir/bsd/lhost-deutschetelekom-02.eml +68 -0
- data/set-of-emails/maildir/bsd/lhost-deutschetelekom-03.eml +50 -0
- data/set-of-emails/should-not-crash/p5-664-iomart-mail-filter.eml +258 -0
- data/set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet/.gitkeep +0 -0
- metadata +8 -2
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: 701bb8a21375a15d4189cc3a3f8929884ab3406483526d009bb1e257b610cf9b
|
|
4
|
+
data.tar.gz: 2c319b65b0f72c9351dc099877505b6176c3b980e813fc10e53e67be504cafe1
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
6
|
+
metadata.gz: a9458959a372839a8626a31bf487718338a4942c5a27e9d99e4b7efac8e340263c4dab88c8988a197e3b1e72be15b50cf69fcc07c41b6c2f007d03fba33854b1
|
|
7
|
+
data.tar.gz: 5b3c330b2297749b8d4a91d68e8ef898993457129b87fab8e7222efc0c6b0a4251742bc5330ab22077dc915fc786e96acf511e41ac3bf74775ae8664da3aeabf
|
|
@@ -15,8 +15,6 @@ jobs:
|
|
|
15
15
|
steps:
|
|
16
16
|
- name: Checkout the repository(CRuby)
|
|
17
17
|
uses: actions/checkout@v4
|
|
18
|
-
with:
|
|
19
|
-
ref: ${{ github.event.pull_request.head.ref }}
|
|
20
18
|
- name: Setup CRuby
|
|
21
19
|
uses: ruby/setup-ruby@v1
|
|
22
20
|
with:
|
|
@@ -41,8 +39,6 @@ jobs:
|
|
|
41
39
|
steps:
|
|
42
40
|
- name: Checkout the repository(JRuby)
|
|
43
41
|
uses: actions/checkout@v4
|
|
44
|
-
with:
|
|
45
|
-
ref: ${{ github.event.pull_request.head.ref }}
|
|
46
42
|
- name: Setup JRuby
|
|
47
43
|
uses: ruby/setup-ruby@v1
|
|
48
44
|
with:
|
data/ChangeLog.md
CHANGED
|
@@ -4,6 +4,38 @@ RELEASE NOTES for Ruby version of Sisimai
|
|
|
4
4
|
- download: "https://rubygems.org/gems/sisimai"
|
|
5
5
|
- document: "https://libsisimai.org/"
|
|
6
6
|
|
|
7
|
+
v5.7.0
|
|
8
|
+
---------------------------------------------------------------------------------------------------
|
|
9
|
+
- release: "Mon, 22 Jun 2026 16:22:22 +0900 (JST)"
|
|
10
|
+
- version: "5.7.0"
|
|
11
|
+
- changes:
|
|
12
|
+
- **Bug fixes**
|
|
13
|
+
- #414 Bug fix: The value of command should be `RCPT` when `RCPT first` in the error message.
|
|
14
|
+
- #433 Check whether `lowerchunk` is nil or not to avoid `NoMethodError`, `SystemStackError`
|
|
15
|
+
in `haircut` method of `Sisimai::RFC2045`. #434 #436 Thanks to @SAY-5.
|
|
16
|
+
- **MTA modules and error message patterns**
|
|
17
|
+
- #410 #412 Improvement in error message patterns.
|
|
18
|
+
- Update error message patterns in `AuthFailure`, `ContentError`, `NoRelaying`, `Rejected`,
|
|
19
|
+
`UserUnknown`, `SystemError`, `Suspend`, and `Sisimai::Rhost::Microsoft`.
|
|
20
|
+
- Remove error message patterns already defined in `Sisimai::Reason` from some MTA moudles of
|
|
21
|
+
`Sisimai::Lhost`.
|
|
22
|
+
- Move error message patterns from some MTA modules of `Sisimai::Lhost` to `Sisimai::Reason`.
|
|
23
|
+
- #413 List all the Zoho domains.
|
|
24
|
+
- Update the order of `ClassOrder` in `Sisimai::Reason`.
|
|
25
|
+
- #421 Code improvement for detecting bounce messages returned from Google Groups.
|
|
26
|
+
- #437 Implement `Sisimai::Lhost::DeutscheTelekom` to decode bounce messages generated by Smail
|
|
27
|
+
3 or Deutsche Telekom.
|
|
28
|
+
- **Code improvements and Environment**
|
|
29
|
+
- **#430 Sisimai does not support Ruby 4.0.0 until the end of 2027.**
|
|
30
|
+
- #417 Use a switch statement instead of if-else for better readability.
|
|
31
|
+
- #423 Fix and update comments in `Sisimai::SMTP::Status`.
|
|
32
|
+
- #428 Remove useless code blocks.
|
|
33
|
+
- **EXPERIMENTAL Features**
|
|
34
|
+
- #424 #425 Change the data type of `Toxic` field implemented at v5.5.0 from Boolean to Integer
|
|
35
|
+
with `-1` as the default value to allow score-based evaluation of recipient address toxicity.
|
|
36
|
+
- #426 #427 Implement `Bogus` field as an Integer with `-1` as the default value to record the
|
|
37
|
+
unreliability score of a bounce message.
|
|
38
|
+
|
|
7
39
|
v5.6.0
|
|
8
40
|
---------------------------------------------------------------------------------------------------
|
|
9
41
|
- release: "Mon, 2 Feb 2026 18:30:22 +0900 (JST)"
|
data/README-JA.md
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|

|
|
2
|
-
[](https://github.com/sisimai/rb-sisimai/blob/
|
|
2
|
+
[](https://github.com/sisimai/rb-sisimai/blob/5-stable/LICENSE)
|
|
3
3
|
[](https://www.ruby-lang.org/)
|
|
4
4
|
[](https://badge.fury.io/rb/sisimai)
|
|
5
5
|
[](https://codecov.io/github/sisimai/rb-sisimai)
|
|
@@ -65,14 +65,15 @@ Sisimai(シシマイ)は複雑で多種多様なバウンスメールを解析
|
|
|
65
65
|
The key features of Sisimai
|
|
66
66
|
---------------------------------------------------------------------------------------------------
|
|
67
67
|
* __バウンスメールを構造化したデータに変換__
|
|
68
|
-
* 以下
|
|
68
|
+
* 以下28項目の情報を含むデータ構造[^2]
|
|
69
69
|
* __基本的情報__: `timestamp`, `origin`
|
|
70
70
|
* __発信者情報__: `addresser`, `senderdomain`,
|
|
71
71
|
* __受信者情報__: `recipient`, `destination`, `alias`
|
|
72
72
|
* __配信の情報__: `action`, `replycode`, `deliverystatus`, `command`
|
|
73
73
|
* __エラー情報__: `reason`, `diagnosticcode`, `diagnostictype`, `feedbacktype`, `hardbounce`
|
|
74
74
|
* __メール情報__: `subject`, `messageid`, `listid`,
|
|
75
|
-
* __
|
|
75
|
+
* __評価用項目__: `toxic`, `bogus`, `catch`
|
|
76
|
+
* __その他情報__: `decodedby`, `timezoneoffset`, `lhost`, `rhost`, `token`
|
|
76
77
|
* __出力可能な形式__
|
|
77
78
|
* Ruby (Hash, Array)
|
|
78
79
|
* JSON
|
|
@@ -83,9 +84,9 @@ The key features of Sisimai
|
|
|
83
84
|
* `gem install`
|
|
84
85
|
* `git clone && make`
|
|
85
86
|
* __高い解析精度__
|
|
86
|
-
* [
|
|
87
|
+
* [61種類のMTAs/MDAs/ESPs](https://libsisimai.org/en/engine/)に対応
|
|
87
88
|
* Feedback Loop(ARF)にも対応
|
|
88
|
-
* [
|
|
89
|
+
* [34種類のバウンス理由](https://libsisimai.org/en/reason/)を検出
|
|
89
90
|
|
|
90
91
|
[^2]: コールバック機能を使用すると`catch`アクセサの下に独自のデータを追加できます
|
|
91
92
|
|
|
@@ -115,10 +116,10 @@ Install
|
|
|
115
116
|
### From RubyGems.org
|
|
116
117
|
```shell
|
|
117
118
|
$ sudo gem install sisimai
|
|
118
|
-
Fetching: sisimai-5.
|
|
119
|
-
Successfully installed sisimai-5.
|
|
120
|
-
Parsing documentation for sisimai-5.
|
|
121
|
-
Installing ri documentation for sisimai-5.
|
|
119
|
+
Fetching: sisimai-5.7.0.gem (100%)
|
|
120
|
+
Successfully installed sisimai-5.7.0
|
|
121
|
+
Parsing documentation for sisimai-5.7.0
|
|
122
|
+
Installing ri documentation for sisimai-5.7.0
|
|
122
123
|
Done installing documentation for sisimai after 6 seconds
|
|
123
124
|
1 gem installed
|
|
124
125
|
```
|
|
@@ -146,13 +147,13 @@ if [ -d "/usr/local/jr" ]; then \
|
|
|
146
147
|
...
|
|
147
148
|
3 gems installed
|
|
148
149
|
/opt/local/bin/rake install
|
|
149
|
-
sisimai 5.
|
|
150
|
-
sisimai (5.
|
|
150
|
+
sisimai 5.7.0 built to pkg/sisimai-5.7.0.gem.
|
|
151
|
+
sisimai (5.7.0) installed.
|
|
151
152
|
if [ -d "/usr/local/jr" ]; then \
|
|
152
153
|
PATH="/usr/local/jr/bin:$PATH" /usr/local/jr/bin/rake install; \
|
|
153
154
|
fi
|
|
154
|
-
sisimai 5.
|
|
155
|
-
sisimai (5.
|
|
155
|
+
sisimai 5.7.0 built to pkg/sisimai-5.7.0-java.gem.
|
|
156
|
+
sisimai (5.7.0) installed.
|
|
156
157
|
```
|
|
157
158
|
|
|
158
159
|
Usage
|
|
@@ -333,7 +334,8 @@ Output example
|
|
|
333
334
|
"timezoneoffset": "+0900",
|
|
334
335
|
"replycode": 550,
|
|
335
336
|
"token": "84656774898baa90660be3e12fe0526e108d4473",
|
|
336
|
-
"
|
|
337
|
+
"bogus": -1,
|
|
338
|
+
"toxic": -1,
|
|
337
339
|
"diagnostictype": "SMTP",
|
|
338
340
|
"timestamp": 1650119685,
|
|
339
341
|
"diagnosticcode": "host gmail-smtp-in.l.google.com[64.233.187.27] said: This mail has been blocked because the sender is unauthenticated. Gmail requires all senders to authenticate with either SPF or DKIM. Authentication results: DKIM = did not pass SPF [relay3.example.com] with ip: [192.0.2.22] = did not pass For instructions on setting up authentication, go to https://support.google.com/mail/answer/81126#authentication c2-202200202020202020222222cat.127 - gsmtp (in reply to end of DATA command)",
|
|
@@ -405,6 +407,7 @@ Sisimai 5で3個のESPモジュール名(解析エンジン)が変更になり
|
|
|
405
407
|
| Zoho (added at v5.5.0) | なし | `Rhost::Zoho` |
|
|
406
408
|
| DragonFly Mail Agent (added at v5.1.0) | なし | `Lhost::DragonFly` |
|
|
407
409
|
| Mimecast (added at v5.5.0) | なし | `Lhost::Mimecast` |
|
|
410
|
+
| DeutscheTelekom (added at v5.7.0) | なし | `Lhost::DeutscheTelekom` |
|
|
408
411
|
|
|
409
412
|
Bounce Reasons
|
|
410
413
|
---------------------------------------------------------------------------------------------------
|
|
@@ -436,7 +439,7 @@ Bug report
|
|
|
436
439
|
Emails could not be decoded
|
|
437
440
|
---------------------------------------------------------------------------------------------------
|
|
438
441
|
Sisimaiで解析できないバウンスメールは
|
|
439
|
-
[set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet](https://github.com/sisimai/set-of-emails/
|
|
442
|
+
[set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet](https://github.com/sisimai/rb-sisimai/tree/5-stable/set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet)ディレクトリに追加してPull-Requestを送ってください。
|
|
440
443
|
|
|
441
444
|
Other Information
|
|
442
445
|
===================================================================================================
|
|
@@ -453,7 +456,7 @@ Related sites
|
|
|
453
456
|
|
|
454
457
|
See also
|
|
455
458
|
---------------------------------------------------------------------------------------------------
|
|
456
|
-
* [README.md - README.md in English(🇬🇧)](https://github.com/sisimai/rb-sisimai/blob/
|
|
459
|
+
* [README.md - README.md in English(🇬🇧)](https://github.com/sisimai/rb-sisimai/blob/5-stable/README.md)
|
|
457
460
|
* [RFC3463 - Enhanced Mail System Status Codes](https://tools.ietf.org/html/rfc3463)
|
|
458
461
|
* [RFC3464 - An Extensible Message Format for Delivery Status Notifications](https://tools.ietf.org/html/rfc3464)
|
|
459
462
|
* [RFC3834 - Recommendations for Automatic Responses to Electronic Mail](https://tools.ietf.org/html/rfc3834)
|
data/README.md
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|

|
|
2
|
-
[](https://github.com/sisimai/rb-sisimai/blob/
|
|
2
|
+
[](https://github.com/sisimai/rb-sisimai/blob/5-stable/LICENSE)
|
|
3
3
|
[](https://www.ruby-lang.org/)
|
|
4
4
|
[](https://badge.fury.io/rb/sisimai)
|
|
5
5
|
[](https://codecov.io/github/sisimai/rb-sisimai)
|
|
@@ -67,14 +67,15 @@ of Sisimai is ported from [the Perl version of Sisimai](https://github.com/sisim
|
|
|
67
67
|
The key features of Sisimai
|
|
68
68
|
---------------------------------------------------------------------------------------------------
|
|
69
69
|
* __Decode email bounces to structured data__
|
|
70
|
-
* Sisimai provides detailed insights into bounce emails by extracting
|
|
70
|
+
* Sisimai provides detailed insights into bounce emails by extracting 28 key data points.[^2]
|
|
71
71
|
* __Essential information__: `timestamp`, `origin`
|
|
72
72
|
* __Sender information__: `addresser`, `senderdomain`,
|
|
73
73
|
* __Recipient information__: `recipient`, `destination`, `alias`
|
|
74
74
|
* __Delivery information__: `action`, `replycode`, `deliverystatus`, `command`
|
|
75
75
|
* __Bounce details__: `reason`, `diagnosticcode`, `diagnostictype`, `feedbacktype`, `feedbackid`, `hardbounce`
|
|
76
76
|
* __Message details__: `subject`, `messageid`, `listid`,
|
|
77
|
-
*
|
|
77
|
+
* __Evaluation metrics (User-calculated)__: `toxic`, `bogus`, `catch`
|
|
78
|
+
* __Additional information__: `decodedby`, `timezoneoffset`, `lhost`, `rhost`, `token`
|
|
78
79
|
* Output formats
|
|
79
80
|
* Ruby (Hash, Array)
|
|
80
81
|
* JSON
|
|
@@ -85,7 +86,7 @@ The key features of Sisimai
|
|
|
85
86
|
* `gem install`
|
|
86
87
|
* `git clone && make`
|
|
87
88
|
* __High Precision of Analysis__
|
|
88
|
-
* Support [
|
|
89
|
+
* Support [61 MTAs/MDAs/ESPs](https://libsisimai.org/en/engine/)
|
|
89
90
|
* Support Feedback Loop Message(ARF)
|
|
90
91
|
* Can detect [34 bounce reasons](https://libsisimai.org/en/reason/)
|
|
91
92
|
|
|
@@ -114,10 +115,10 @@ Install
|
|
|
114
115
|
### From RubyGems
|
|
115
116
|
```shell
|
|
116
117
|
$ sudo gem install sisimai
|
|
117
|
-
Fetching: sisimai-5.
|
|
118
|
-
Successfully installed sisimai-5.
|
|
119
|
-
Parsing documentation for sisimai-5.
|
|
120
|
-
Installing ri documentation for sisimai-5.
|
|
118
|
+
Fetching: sisimai-5.7.0.gem (100%)
|
|
119
|
+
Successfully installed sisimai-5.7.0
|
|
120
|
+
Parsing documentation for sisimai-5.7.0
|
|
121
|
+
Installing ri documentation for sisimai-5.7.0
|
|
121
122
|
Done installing documentation for sisimai after 6 seconds
|
|
122
123
|
1 gem installed
|
|
123
124
|
```
|
|
@@ -145,13 +146,13 @@ if [ -d "/usr/local/jr" ]; then \
|
|
|
145
146
|
...
|
|
146
147
|
3 gems installed
|
|
147
148
|
/opt/local/bin/rake install
|
|
148
|
-
sisimai 5.
|
|
149
|
-
sisimai (5.
|
|
149
|
+
sisimai 5.7.0 built to pkg/sisimai-5.7.0.gem.
|
|
150
|
+
sisimai (5.7.0) installed.
|
|
150
151
|
if [ -d "/usr/local/jr" ]; then \
|
|
151
152
|
PATH="/usr/local/jr/bin:$PATH" /usr/local/jr/bin/rake install; \
|
|
152
153
|
fi
|
|
153
|
-
sisimai 5.
|
|
154
|
-
sisimai (5.
|
|
154
|
+
sisimai 5.7.0 built to pkg/sisimai-5.7.0-java.gem.
|
|
155
|
+
sisimai (5.7.0) installed.
|
|
155
156
|
```
|
|
156
157
|
|
|
157
158
|
Usage
|
|
@@ -332,7 +333,8 @@ Output example
|
|
|
332
333
|
"timezoneoffset": "+0900",
|
|
333
334
|
"replycode": 550,
|
|
334
335
|
"token": "84656774898baa90660be3e12fe0526e108d4473",
|
|
335
|
-
"
|
|
336
|
+
"bogus": -1,
|
|
337
|
+
"toxic": -1,
|
|
336
338
|
"diagnostictype": "SMTP",
|
|
337
339
|
"timestamp": 1650119685,
|
|
338
340
|
"diagnosticcode": "host gmail-smtp-in.l.google.com[64.233.187.27] said: This mail has been blocked because the sender is unauthenticated. Gmail requires all senders to authenticate with either SPF or DKIM. Authentication results: DKIM = did not pass SPF [relay3.example.com] with ip: [192.0.2.22] = did not pass For instructions on setting up authentication, go to https://support.google.com/mail/answer/81126#authentication c2-202200202020202020222222cat.127 - gsmtp (in reply to end of DATA command)",
|
|
@@ -356,12 +358,12 @@ Beginning with v5.5.0, Sisimai requires **Ruby 2.5.0 or later.**
|
|
|
356
358
|
| System requirements (CRuby) | 2.1 - 3.3.0 | **2.5** or later |
|
|
357
359
|
| System requirements (JRuby) | 9.0.4.0 - 9.1.17.0 | **9.2** or later |
|
|
358
360
|
| Callback feature for the original email file | N/A | Available[^3] |
|
|
359
|
-
| The number of MTA/ESP modules | 68 |
|
|
361
|
+
| The number of MTA/ESP modules | 68 | 61 |
|
|
360
362
|
| The number of detectable bounce reasons | 29 | 34 |
|
|
361
363
|
| Dependencies (Except Ruby Standard Gems) | 1 gem | 1 gem |
|
|
362
|
-
| Source lines of code | 10,300 lines | 9,
|
|
364
|
+
| Source lines of code | 10,300 lines | 9,500 lines |
|
|
363
365
|
| Test frameworks | rspec | minitest |
|
|
364
|
-
| The number of tests in spec/ or test/ directory | 311,000 tests |
|
|
366
|
+
| The number of tests in spec/ or test/ directory | 311,000 tests | 255,000 tests |
|
|
365
367
|
| The number of bounce emails decoded/sec (CRuby)[^4] | 620 emails | 620 emails |
|
|
366
368
|
| License | 2 Clause BSD | 2 Caluse BSD |
|
|
367
369
|
| Commercial support | Available | Available |
|
|
@@ -406,6 +408,7 @@ available at [LIBSISIMAI.ORG/EN/ENGINE](https://libsisimai.org/en/engine/)
|
|
|
406
408
|
| Zoho (added at v5.5.0) | None | `Rhost::Zoho` |
|
|
407
409
|
| DragonFly Mail Agent (added at v5.1.0) | None | `Lhost::DragonFly` |
|
|
408
410
|
| Mimecast (added at v5.5.0) | None | `Lhost::Mimecast` |
|
|
411
|
+
| DeutscheTelekom (added at v5.7.0) | None | `Lhost::DeutscheTelekom` |
|
|
409
412
|
|
|
410
413
|
Bounce Reasons
|
|
411
414
|
---------------------------------------------------------------------------------------------------
|
|
@@ -436,10 +439,9 @@ Please use the [issue tracker](https://github.com/sisimai/rb-sisimai/issues) to
|
|
|
436
439
|
|
|
437
440
|
Emails could not be decoded
|
|
438
441
|
---------------------------------------------------------------------------------------------------
|
|
439
|
-
Bounce mails which could not be decoded by Sisimai are saved in
|
|
440
|
-
[set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet](https://github.com/sisimai/set-of-emails/
|
|
441
|
-
|
|
442
|
-
directory and send Pull-Request to this repository.
|
|
442
|
+
Bounce mails which could not be decoded by Sisimai are saved in
|
|
443
|
+
[set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet](https://github.com/sisimai/rb-sisimai/tree/5-stable/set-of-emails/to-be-debugged-because/sisimai-cannot-parse-yet) directory. If you have found any bounce email cannot be decoded
|
|
444
|
+
using Sisimai, please add the email into the directory and send Pull-Request to this repository.
|
|
443
445
|
|
|
444
446
|
Other Information
|
|
445
447
|
===================================================================================================
|
|
@@ -456,7 +458,7 @@ Related Sites
|
|
|
456
458
|
|
|
457
459
|
See also
|
|
458
460
|
---------------------------------------------------------------------------------------------------
|
|
459
|
-
* [README-JA.md - README.md in Japanese(🇯🇵)](https://github.com/sisimai/rb-sisimai/blob/
|
|
461
|
+
* [README-JA.md - README.md in Japanese(🇯🇵)](https://github.com/sisimai/rb-sisimai/blob/5-stable/README-JA.md)
|
|
460
462
|
* [RFC3463 - Enhanced Mail System Status Codes](https://tools.ietf.org/html/rfc3463)
|
|
461
463
|
* [RFC3464 - An Extensible Message Format for Delivery Status Notifications](https://tools.ietf.org/html/rfc3464)
|
|
462
464
|
* [RFC3834 - Recommendations for Automatic Responses to Electronic Mail](https://tools.ietf.org/html/rfc3834)
|
data/lib/sisimai/address.rb
CHANGED
|
@@ -147,8 +147,8 @@ module Sisimai
|
|
|
147
147
|
# Check each characters
|
|
148
148
|
if Delimiters[e]
|
|
149
149
|
# The character is a delimiter character
|
|
150
|
-
|
|
151
|
-
|
|
150
|
+
case e
|
|
151
|
+
when "," # Separator of email addresses or not
|
|
152
152
|
if v[:address].start_with?('<') && v[:address].end_with?('>') && v[:address].include?('@')
|
|
153
153
|
# An email address has already been picked
|
|
154
154
|
if readcursor & Indicators[:'comment-block'] > 0
|
|
@@ -168,11 +168,7 @@ module Sisimai
|
|
|
168
168
|
# "Neko, Nyaan" <neko@nyaan.example.org> OR <"neko,nyaan"@example.org>
|
|
169
169
|
p.empty? ? (v[:name] += e) : (v[p] += e)
|
|
170
170
|
end
|
|
171
|
-
|
|
172
|
-
end # End of if(',')
|
|
173
|
-
|
|
174
|
-
if e == '<'
|
|
175
|
-
# <: The beginning of an email address or not
|
|
171
|
+
when "<" # <: The beginning of an email address or not
|
|
176
172
|
if v[:address].size > 0
|
|
177
173
|
p.empty? ? (v[:name] += e) : (v[p] += e)
|
|
178
174
|
else
|
|
@@ -181,12 +177,7 @@ module Sisimai
|
|
|
181
177
|
v[:address] += e
|
|
182
178
|
p = :address
|
|
183
179
|
end
|
|
184
|
-
|
|
185
|
-
end
|
|
186
|
-
# End of if('<')
|
|
187
|
-
|
|
188
|
-
if e == '>'
|
|
189
|
-
# >: The end of an email address or not
|
|
180
|
+
when ">" # >: The end of an email address or not
|
|
190
181
|
if readcursor & Indicators[:'email-address'] > 0
|
|
191
182
|
# <neko@example.org>
|
|
192
183
|
readcursor &= ~Indicators[:'email-address']
|
|
@@ -196,11 +187,7 @@ module Sisimai
|
|
|
196
187
|
# a comment block or a display name
|
|
197
188
|
p.empty? ? (v[:name] == e) : (v[:comment] -= e)
|
|
198
189
|
end
|
|
199
|
-
|
|
200
|
-
end # End of if('>')
|
|
201
|
-
|
|
202
|
-
if e == '('
|
|
203
|
-
# The beginning of a comment block or not
|
|
190
|
+
when "(" # The beginning of a comment block or not
|
|
204
191
|
if readcursor & Indicators[:'email-address'] > 0
|
|
205
192
|
# <"neko(nyaan)"@example.org> or <neko(nyaan)@example.org>
|
|
206
193
|
if v[:address].include?('"')
|
|
@@ -228,11 +215,7 @@ module Sisimai
|
|
|
228
215
|
v[:comment] += e
|
|
229
216
|
p = :comment
|
|
230
217
|
end
|
|
231
|
-
|
|
232
|
-
end # End of if('(')
|
|
233
|
-
|
|
234
|
-
if e == ')'
|
|
235
|
-
# The end of a comment block or not
|
|
218
|
+
when ")" # The end of a comment block or not
|
|
236
219
|
if readcursor & Indicators[:'email-address'] > 0
|
|
237
220
|
# <"neko(nyaan)"@example.org> OR <neko(nyaan)@example.org>
|
|
238
221
|
if v[:address].include?('"')
|
|
@@ -255,11 +238,7 @@ module Sisimai
|
|
|
255
238
|
v[:name] += e
|
|
256
239
|
p = ''
|
|
257
240
|
end
|
|
258
|
-
|
|
259
|
-
end # End of if(')')
|
|
260
|
-
|
|
261
|
-
if e == '"'
|
|
262
|
-
# The beginning or the end of a quoted-string
|
|
241
|
+
when '"' # The beginning or the end of a quoted-string
|
|
263
242
|
if p.size > 0
|
|
264
243
|
# email-address or comment-block
|
|
265
244
|
v[p] += e
|
|
@@ -271,12 +250,10 @@ module Sisimai
|
|
|
271
250
|
readcursor &= ~Indicators[:'quoted-string']
|
|
272
251
|
p = ''
|
|
273
252
|
end
|
|
274
|
-
|
|
275
|
-
end # End of if('"')
|
|
253
|
+
end # End of case-when
|
|
276
254
|
else
|
|
277
255
|
# The character is not a delimiter
|
|
278
256
|
p.empty? ? (v[:name] += e) : (v[p] += e)
|
|
279
|
-
next
|
|
280
257
|
end
|
|
281
258
|
end
|
|
282
259
|
|
data/lib/sisimai/arf.rb
CHANGED
|
@@ -193,7 +193,7 @@ module Sisimai
|
|
|
193
193
|
# X-Apple-Unsubscribe: true
|
|
194
194
|
last if mhead["x-apple-unsubscribe"] != "true" || mhead["from"].include?('@') == false
|
|
195
195
|
dscontents[0]["recipient"] = mhead["from"]
|
|
196
|
-
dscontents[0]["diagnosis"] =
|
|
196
|
+
dscontents[0]["diagnosis"] = emailparts[0]
|
|
197
197
|
dscontents[0]["feedbacktype"] = "opt-out"
|
|
198
198
|
|
|
199
199
|
# Addpend To: field as a pseudo header
|
|
@@ -214,14 +214,12 @@ module Sisimai
|
|
|
214
214
|
end
|
|
215
215
|
return nil if recipients == 0
|
|
216
216
|
|
|
217
|
-
anotherone = ": #{
|
|
218
|
-
anotherone = anotherone.chop if anotherone[-1, 1] == ","
|
|
219
|
-
|
|
217
|
+
anotherone = ": #{anotherone.chop}" if anotherone != ""
|
|
220
218
|
j = -1
|
|
221
219
|
dscontents.each do |e|
|
|
222
220
|
# Tidy up the error message in e.Diagnosis, Try to detect the bounce reason.
|
|
223
221
|
j += 1
|
|
224
|
-
e["diagnosis"] =
|
|
222
|
+
e["diagnosis"] = e["diagnosis"] + anotherone
|
|
225
223
|
e["reason"] = "feedback"
|
|
226
224
|
e["rhost"] = remotehost
|
|
227
225
|
e["lhost"] = reportedby
|
data/lib/sisimai/fact.rb
CHANGED
|
@@ -20,6 +20,7 @@ module Sisimai
|
|
|
20
20
|
:action, # [String] The value of Action: header
|
|
21
21
|
:addresser, # [Sisimai::Address] From address
|
|
22
22
|
:alias, # [String] Alias of the recipient address
|
|
23
|
+
:bogus, # [Integer] EXPERIMENTAL
|
|
23
24
|
:catch, # [?] Results generated by hook method
|
|
24
25
|
:command, # [String] The last SMTP command
|
|
25
26
|
:decodedby, # [String] MTA module name since v5.2.0
|
|
@@ -43,7 +44,7 @@ module Sisimai
|
|
|
43
44
|
:timestamp, # [Sisimai::Time] Date: header in the original message
|
|
44
45
|
:timezoneoffset, # [Integer] Time zone offset(seconds)
|
|
45
46
|
:token, # [String] Message token/MD5 Hex digest value
|
|
46
|
-
:toxic, # [
|
|
47
|
+
:toxic, # [Integer] EXPERIMENTAL
|
|
47
48
|
]
|
|
48
49
|
attr_accessor(*@@rwaccessors)
|
|
49
50
|
|
|
@@ -71,6 +72,7 @@ module Sisimai
|
|
|
71
72
|
@alias = argvs['alias'] || ''
|
|
72
73
|
@addresser = argvs['addresser']
|
|
73
74
|
@action = argvs['action']
|
|
75
|
+
@bogus = argvs['bogus']
|
|
74
76
|
@catch = argvs['catch']
|
|
75
77
|
@command = argvs['command']
|
|
76
78
|
@decodedby = argvs['decodedby']
|
|
@@ -141,7 +143,6 @@ module Sisimai
|
|
|
141
143
|
"replycode" => e["replycode"],
|
|
142
144
|
"rhost" => e["rhost"],
|
|
143
145
|
"decodedby" => e["agent"],
|
|
144
|
-
"toxic" => e["toxic"],
|
|
145
146
|
}
|
|
146
147
|
|
|
147
148
|
# EMAILADDRESS: Detect an email address from message/rfc822 part
|
|
@@ -344,7 +345,7 @@ module Sisimai
|
|
|
344
345
|
p1 = dc.index('<html>')
|
|
345
346
|
p2 = dc.index('</html>')
|
|
346
347
|
piece['diagnosticcode'][p1, p2 + 7 - p1] = '' if p1 && p2
|
|
347
|
-
piece['diagnosticcode'] =
|
|
348
|
+
piece['diagnosticcode'] = piece['diagnosticcode'].split.join(" ")
|
|
348
349
|
end
|
|
349
350
|
|
|
350
351
|
if Sisimai::String.is_8bit(piece['diagnosticcode'])
|
|
@@ -357,8 +358,12 @@ module Sisimai
|
|
|
357
358
|
piece["diagnostictype"] = "SMTP" if %w[feedback vacation].include?(piece["reason"]) == false
|
|
358
359
|
end
|
|
359
360
|
|
|
360
|
-
#
|
|
361
|
+
# When "RCPT first" in the error message, set "RCPT" as the last command.
|
|
362
|
+
# - <<< 503 RCPT first (#5.5.1)
|
|
363
|
+
# - <<< 503-5.5.1 RCPT first. A mail transaction protocol command was issued ...
|
|
364
|
+
# - RCPT first (in reply to DATA command)
|
|
361
365
|
piece['command'] = '' if Sisimai::SMTP::Command.test(piece['command']) == false
|
|
366
|
+
piece['command'] = 'RCPT' if piece['diagnosticcode'].include?('RCPT first')
|
|
362
367
|
|
|
363
368
|
# Create parameters for the constructor
|
|
364
369
|
as = Sisimai::Address.new(piece['addresser']) || next; next if as.void
|
|
@@ -379,13 +384,14 @@ module Sisimai
|
|
|
379
384
|
ea.each { |q| thing[q] = piece[q] if thing[q].nil? || thing[q].empty? }
|
|
380
385
|
|
|
381
386
|
# Other accessors
|
|
387
|
+
thing['bogus'] = 0
|
|
382
388
|
thing['catch'] = piece['catch'] || nil
|
|
383
389
|
thing["feedbackid"] = ""
|
|
384
390
|
thing['hardbounce'] = piece['hardbounce']
|
|
385
|
-
thing['toxic'] = piece['toxic']
|
|
386
391
|
thing['replycode'] = Sisimai::SMTP::Reply.find(piece['diagnosticcode']) if thing['replycode'].empty?
|
|
387
392
|
thing['timestamp'] = TimeModule.parse(::Time.at(piece['timestamp']).to_s)
|
|
388
393
|
thing['timezoneoffset'] = piece['timezoneoffset'] || '+0000'
|
|
394
|
+
thing['toxic'] = 0
|
|
389
395
|
ea.each { |q| thing[q] = piece[q] if thing[q].empty? }
|
|
390
396
|
|
|
391
397
|
# ALIAS
|
|
@@ -483,44 +489,12 @@ module Sisimai
|
|
|
483
489
|
|
|
484
490
|
# Feedback-ID: 1.us-west-2.QHuyeCQrGtIIMGKQfVdUhP9hCQR2LglVOrRamBc+Prk=:AmazonSES
|
|
485
491
|
thing["feedbackid"] = rfc822data["feedback-id"] || ""
|
|
486
|
-
thing["toxic"] ||= is_toxic(thing)
|
|
487
492
|
|
|
488
493
|
listoffact << Sisimai::Fact.new(thing)
|
|
489
494
|
end
|
|
490
495
|
return listoffact
|
|
491
496
|
end
|
|
492
497
|
|
|
493
|
-
def self.is_toxic(thing = nil)
|
|
494
|
-
return false unless thing
|
|
495
|
-
cr = thing['reason'] || 'undefined'
|
|
496
|
-
cv = thing['replycode'] || ''
|
|
497
|
-
cw = thing['deliverystatus'] || ''
|
|
498
|
-
|
|
499
|
-
# 1. Hard bounces or some soft bounces with a permanent error.
|
|
500
|
-
# 1-1. Hard bounce: UserUnknown, HostUnknown, HasMoved, NotAccept
|
|
501
|
-
# 1-2. Almost hard bounce: Suspend, Suppressed
|
|
502
|
-
return false if cv.start_with?('4') || cw.start_with?('4')
|
|
503
|
-
return true if %w[userunknown hostunknown hasmoved notaccept suspend suppressed].any? { |a| cr == a }
|
|
504
|
-
|
|
505
|
-
if %w[mailboxfull filtered norelaying].any? { |a| cr == a }
|
|
506
|
-
# 2. Several softbounces: MailboxFull, Filtered, NoRelaying
|
|
507
|
-
# 2-1. The SMTP command is "RCPT" except "MailboxFull".
|
|
508
|
-
# 2-2. The SMTP reply code begins with "5" such as "550".
|
|
509
|
-
# 2-3. The SMTP status code is explicit code (not empty, not 5.9.***).
|
|
510
|
-
# 2-4. The SMTP status code begins with "5." such as "5.1.1".
|
|
511
|
-
return true if cr != 'mailboxfull' && thing['command'] == "RCPT"
|
|
512
|
-
return true if cv.start_with?('5')
|
|
513
|
-
return false if Sisimai::SMTP::Status.is_explicit(cw) == false
|
|
514
|
-
return true if cw.start_with?('5.')
|
|
515
|
-
|
|
516
|
-
elsif cr == 'feedback'
|
|
517
|
-
# 3. Feedback Loop
|
|
518
|
-
# 3-1. The Feedback Type is any of "abuse", "fraud", "opt-out"
|
|
519
|
-
return true if %w[abuse fraud opt-out].any? { |a| thing['feedbacktype'] == a }
|
|
520
|
-
end
|
|
521
|
-
return false
|
|
522
|
-
end
|
|
523
|
-
|
|
524
498
|
# Create message token from addresser and recipient
|
|
525
499
|
# @param [String] addr1 Sender address
|
|
526
500
|
# @param [String] addr2 Recipient address
|
|
@@ -551,6 +525,7 @@ module Sisimai
|
|
|
551
525
|
stringdata.each { |e| v[e] = self.send(e.to_sym) || '' }
|
|
552
526
|
v['hardbounce'] = self.hardbounce
|
|
553
527
|
v['toxic'] = self.toxic
|
|
528
|
+
v['bogus'] = self.bogus
|
|
554
529
|
v['addresser'] = self.addresser.address
|
|
555
530
|
v['recipient'] = self.recipient.address
|
|
556
531
|
v['timestamp'] = self.timestamp.to_time.to_i
|
|
@@ -63,8 +63,6 @@ module Sisimai::Lhost
|
|
|
63
63
|
end
|
|
64
64
|
end
|
|
65
65
|
return nil if recipients == 0
|
|
66
|
-
|
|
67
|
-
dscontents.each { |e| e['diagnosis'] = Sisimai::String.sweep(e['diagnosis']) }
|
|
68
66
|
return {"ds" => dscontents, "rfc822" => emailparts[1]}
|
|
69
67
|
end
|
|
70
68
|
def description; return 'TransWARE Active!hunter'; end
|
|
@@ -124,7 +124,8 @@ module Sisimai::Lhost
|
|
|
124
124
|
recipients = 0 # (Integer) The number of 'Final-Recipient' header
|
|
125
125
|
whatnotify = jsonobject["notificationType"][0, 1] || ""
|
|
126
126
|
|
|
127
|
-
|
|
127
|
+
case whatnotify
|
|
128
|
+
when "B"
|
|
128
129
|
# "notificationType":"Bounce"
|
|
129
130
|
p = jsonobject["bounce"]
|
|
130
131
|
r = p["bounceType"] == "Permanent" ? "5" : "4"
|
|
@@ -137,7 +138,7 @@ module Sisimai::Lhost
|
|
|
137
138
|
v = dscontents[-1]
|
|
138
139
|
end
|
|
139
140
|
v["recipient"] = e["emailAddress"]
|
|
140
|
-
v["diagnosis"] =
|
|
141
|
+
v["diagnosis"] = e["diagnosticCode"]
|
|
141
142
|
v["command"] = Sisimai::SMTP::Command.find(v["diagnosis"])
|
|
142
143
|
v["action"] = e["action"]
|
|
143
144
|
v["status"] = Sisimai::SMTP::Status.find(v["diagnosis"], r)
|
|
@@ -152,8 +153,7 @@ module Sisimai::Lhost
|
|
|
152
153
|
next if ReasonPair[f] != p["bounceSubType"]
|
|
153
154
|
v["reason"] = f; break
|
|
154
155
|
end
|
|
155
|
-
|
|
156
|
-
elsif whatnotify == "C"
|
|
156
|
+
when "C"
|
|
157
157
|
# "notificationType":"Complaint"
|
|
158
158
|
p = jsonobject["complaint"]
|
|
159
159
|
p["complainedRecipients"].each do |e|
|
|
@@ -170,8 +170,7 @@ module Sisimai::Lhost
|
|
|
170
170
|
v["diagnosis"] = sprintf('"feedbackid":"%s", "useragent":"%s"}', p["feedbackId"], p["userAgent"])
|
|
171
171
|
recipients += 1
|
|
172
172
|
end
|
|
173
|
-
|
|
174
|
-
elsif whatnotify == "D"
|
|
173
|
+
when "D"
|
|
175
174
|
# "notificationType":"Delivery"
|
|
176
175
|
p = jsonobject["delivery"]
|
|
177
176
|
p["recipients"].each do |e|
|
|
@@ -186,13 +185,12 @@ module Sisimai::Lhost
|
|
|
186
185
|
v["action"] = "delivered"
|
|
187
186
|
v["date"] = p["timestamp"]
|
|
188
187
|
v["lhost"] = Sisimai::RFC1123.find(p["reportingMTA"])
|
|
189
|
-
v["diagnosis"] =
|
|
188
|
+
v["diagnosis"] = p["smtpResponse"]
|
|
190
189
|
v["command"] = Sisimai::SMTP::Command.find(v["diagnosis"])
|
|
191
190
|
v["status"] = Sisimai::SMTP::Status.find(v["diagnosis"], "2")
|
|
192
191
|
v["replycode"] = Sisimai::SMTP::Reply.find(v["diagnosis"], "2")
|
|
193
192
|
recipients += 1
|
|
194
193
|
end
|
|
195
|
-
|
|
196
194
|
else
|
|
197
195
|
# Unknown "notificationType" value
|
|
198
196
|
warn sprintf(" ***warning: There is no notificationType field or unknown type of notificationType field")
|
|
@@ -103,7 +103,6 @@ module Sisimai::Lhost
|
|
|
103
103
|
emailparts[1] += sprintf("Subject: %s\n", alternates[3]) if alternates[3] != ""
|
|
104
104
|
end
|
|
105
105
|
|
|
106
|
-
dscontents.each { |e| e["diagnosis"] = Sisimai::String.sweep(e["diagnosis"]) }
|
|
107
106
|
return { "ds" => dscontents, "rfc822" => emailparts[1] }
|
|
108
107
|
end
|
|
109
108
|
def description; return 'Java Apache Mail Enterprise Server'; end
|
|
@@ -11,10 +11,6 @@ module Sisimai::Lhost
|
|
|
11
11
|
message: [' ----- The following addresses had delivery problems -----'],
|
|
12
12
|
error: [' ----- Non-delivered information -----'],
|
|
13
13
|
}.freeze
|
|
14
|
-
MessagesOf = {
|
|
15
|
-
'filtered' => ['Mail Delivery Failed... User unknown'],
|
|
16
|
-
'mailboxfull' => ["The number of messages in recipient's mailbox exceeded the local limit."],
|
|
17
|
-
}.freeze
|
|
18
14
|
|
|
19
15
|
# @asbtract Decodes the bounce message from Biglobe
|
|
20
16
|
# @param [Hash] mhead Message headers of a bounce email
|
|
@@ -76,18 +72,6 @@ module Sisimai::Lhost
|
|
|
76
72
|
end
|
|
77
73
|
end
|
|
78
74
|
return nil if recipients == 0
|
|
79
|
-
|
|
80
|
-
dscontents.each do |e|
|
|
81
|
-
e['diagnosis'] = Sisimai::String.sweep(e['diagnosis'])
|
|
82
|
-
|
|
83
|
-
MessagesOf.each_key do |r|
|
|
84
|
-
# Verify each regular expression of session errors
|
|
85
|
-
next if MessagesOf[r].none? { |a| e['diagnosis'].include?(a) }
|
|
86
|
-
e['reason'] = r
|
|
87
|
-
break
|
|
88
|
-
end
|
|
89
|
-
end
|
|
90
|
-
|
|
91
75
|
return { 'ds' => dscontents, 'rfc822' => emailparts[1] }
|
|
92
76
|
end
|
|
93
77
|
def description; return 'BIGLOBE: https://www.biglobe.ne.jp'; end
|