| 
									
										
										
										
											2017-01-03 09:48:08 -05:00
										 |  |  | USING: accessors arrays assocs calendar calendar.english calendar.format | 
					
						
							| 
									
										
										
										
											2017-01-04 02:50:24 -05:00
										 |  |  | calendar.parser formatting fry grouping io io.crlf io.encodings.ascii | 
					
						
							| 
									
										
										
										
											2017-01-03 09:48:08 -05:00
										 |  |  | io.encodings.binary io.encodings.string io.encodings.utf7 io.encodings.utf8 | 
					
						
							|  |  |  | io.sockets io.sockets.secure io.streams.duplex io.streams.string kernel math | 
					
						
							|  |  |  | math.parser sequences splitting strings ;
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | QUALIFIED: pcre | 
					
						
							|  |  |  | IN: imap | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: imap4-error ind data ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CONSTANT: IMAP4_PORT     143
 | 
					
						
							|  |  |  | CONSTANT: IMAP4_SSL_PORT 993
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! Converts a timestamp to the format imap4 expects. | 
					
						
							|  |  |  | : timestamp>internal-date ( timestamp -- str )
 | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             DD "-" MONTH "-" YYYY " " | 
					
						
							|  |  |  |             hh ":" mm ":" ss " " | 
					
						
							|  |  |  |             [ gmt-offset>> write-gmt-offset ] | 
					
						
							|  |  |  |         } formatted | 
					
						
							|  |  |  |     ] with-string-writer ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : internal-date>timestamp ( str -- timestamp )
 | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         ! Date, month, year. | 
					
						
							|  |  |  |         "-" read-token checked-number | 
					
						
							|  |  |  |         "-" read-token month-abbreviations index 1 +
 | 
					
						
							| 
									
										
										
										
											2018-06-19 20:15:05 -04:00
										 |  |  |         read-sp checked-number spin | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  |         ! Hour, minute second and gmt offset. | 
					
						
							|  |  |  |         read-hms " " expect readln parse-rfc822-gmt-offset <timestamp> | 
					
						
							|  |  |  |     ] with-string-reader  ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 16:14:27 -05:00
										 |  |  | <PRIVATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | : >utf7imap4 ( str -- str' )
 | 
					
						
							|  |  |  |     utf7imap4 encode >string ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : comma-list ( numbers -- str )
 | 
					
						
							|  |  |  |     [ number>string ] map "," join ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : check-status ( ind data -- )
 | 
					
						
							| 
									
										
										
										
											2015-08-13 19:13:05 -04:00
										 |  |  |     over "OK" = not [ imap4-error ] [ 2drop ] if ;
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : read-response-chunk ( stop-expr -- item ? )
 | 
					
						
							|  |  |  |     read-?crlf ascii decode swap dupd pcre:findall | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         dup "^.*{(\\d+)}$" pcre:findall | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |             dup "^\\* (\\d+) [A-Z-]+ (.*)$" pcre:findall | 
					
						
							|  |  |  |             [ ] [ nip first third second ] if-empty
 | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |             ! Literal item to read, such as message body. | 
					
						
							|  |  |  |             nip first second second string>number read ascii decode | 
					
						
							|  |  |  |             read-?crlf drop
 | 
					
						
							|  |  |  |         ] if-empty t
 | 
					
						
							|  |  |  |     ] | 
					
						
							| 
									
										
										
										
											2014-11-29 20:44:43 -05:00
										 |  |  |     [ nip first rest values f ] if-empty ;
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : read-response ( tag -- lines )
 | 
					
						
							|  |  |  |     "^%s (BAD|NO|OK) (.*)$" sprintf | 
					
						
							|  |  |  |     '[ _ read-response-chunk [ suffix ] dip ] { } swap loop
 | 
					
						
							|  |  |  |     unclip-last first2 [ check-status ] keep suffix ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : write-command ( command literal tag -- )
 | 
					
						
							|  |  |  |     -rot [ | 
					
						
							|  |  |  |         [ "%s %s\r\n" sprintf ] [ length "%s %s {%d}\r\n" sprintf ] if-empty
 | 
					
						
							|  |  |  |         ascii encode write flush
 | 
					
						
							|  |  |  |     ] keep [ | 
					
						
							|  |  |  |         read-?crlf drop "\r\n" append write flush
 | 
					
						
							|  |  |  |     ] unless-empty ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : command-response ( command literal -- obj )
 | 
					
						
							|  |  |  |     "ABCD" [ write-command ] [ read-response ] bi ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! Special parsing | 
					
						
							|  |  |  | : parse-items ( seq -- items )
 | 
					
						
							|  |  |  |     first " " split 2 tail ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : parse-list-folders ( str -- folder )
 | 
					
						
							|  |  |  |     "\\* LIST \\(([^\\)]+)\\) \"([^\"]+)\" \"([^\"]+)\"" pcre:findall | 
					
						
							| 
									
										
										
										
											2014-11-29 20:44:43 -05:00
										 |  |  |     first rest values [ utf7imap4 decode ] map ;
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : parse-select-folder ( seq -- count )
 | 
					
						
							|  |  |  |     [ "\\* (\\d+) EXISTS" pcre:findall ] map harvest
 | 
					
						
							|  |  |  |     [ f ] [ first first last last string>number ] if-empty ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! Returns uid if the server supports the UIDPLUS extension. | 
					
						
							|  |  |  | : parse-append-mail ( seq -- uid/f )
 | 
					
						
							|  |  |  |     [ "\\[APPENDUID (\\d+) \\d+\\]" pcre:findall ] map harvest
 | 
					
						
							|  |  |  |     [ f ] [ first first last last string>number ] if-empty ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : parse-status ( seq -- assoc )
 | 
					
						
							|  |  |  |     first "\\* STATUS \"[^\"]+\" \\(([^\\)]+)\\)" pcre:findall first last last
 | 
					
						
							|  |  |  |     " " split 2 group [ string>number ] assoc-map ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-07 18:03:36 -04:00
										 |  |  | : parse-store-mail-line ( str -- pair/f )
 | 
					
						
							|  |  |  |     "\\(FLAGS \\(([^\\)]+)\\) UID (\\d+)\\)" pcre:findall [ f ] [ | 
					
						
							| 
									
										
										
										
											2014-11-29 20:44:43 -05:00
										 |  |  |         first rest values first2 [ " " split ] dip string>number swap 2array
 | 
					
						
							| 
									
										
										
										
											2014-10-07 18:03:36 -04:00
										 |  |  |     ] if-empty ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : parse-store-mail ( seq -- assoc )
 | 
					
						
							|  |  |  |     but-last [ parse-store-mail-line ] map sift ;
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 16:14:27 -05:00
										 |  |  | PRIVATE>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  | ! Constructor | 
					
						
							|  |  |  | : <imap4ssl> ( host -- imap4 )
 | 
					
						
							| 
									
										
										
										
											2016-03-02 18:29:59 -05:00
										 |  |  |     IMAP4_SSL_PORT <inet> f <secure> binary <client> drop
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  |     ! Read the useless welcome message. | 
					
						
							|  |  |  |     dup [ "\\*" read-response drop ] with-stream* ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! IMAP commands | 
					
						
							|  |  |  | : capabilities ( -- caps )
 | 
					
						
							|  |  |  |     "CAPABILITY" "" command-response parse-items ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 16:09:59 -05:00
										 |  |  | : login ( username password -- caps )
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  |     "LOGIN %s \"%s\"" sprintf "" command-response parse-items ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! Folder management | 
					
						
							|  |  |  | : list-folders ( directory -- folders )
 | 
					
						
							|  |  |  |     "LIST \"%s\" *" sprintf "" command-response | 
					
						
							|  |  |  |     but-last [ parse-list-folders ] map ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : select-folder ( mailbox -- count )
 | 
					
						
							|  |  |  |     >utf7imap4 "SELECT \"%s\"" sprintf "" command-response | 
					
						
							|  |  |  |     parse-select-folder ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : create-folder ( mailbox -- )
 | 
					
						
							|  |  |  |     >utf7imap4 "CREATE \"%s\"" sprintf "" command-response | 
					
						
							|  |  |  |     drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : delete-folder ( mailbox -- )
 | 
					
						
							|  |  |  |     >utf7imap4 "DELETE \"%s\"" sprintf "" command-response | 
					
						
							|  |  |  |     drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : rename-folder ( old-name new-name -- )
 | 
					
						
							|  |  |  |     [ >utf7imap4 ] bi@ "RENAME \"%s\" \"%s\"" sprintf "" command-response | 
					
						
							|  |  |  |     drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : status-folder ( mailbox keys -- assoc )
 | 
					
						
							|  |  |  |     [ >utf7imap4 ] dip " " join "STATUS \"%s\" (%s)" sprintf | 
					
						
							|  |  |  |     "" command-response parse-status ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : close-folder ( -- )
 | 
					
						
							|  |  |  |     "CLOSE" "" command-response drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ! Mail management | 
					
						
							|  |  |  | : search-mails ( data-spec str -- uids )
 | 
					
						
							|  |  |  |     [ "UID SEARCH CHARSET UTF-8 %s" sprintf ] dip utf8 encode | 
					
						
							|  |  |  |     command-response parse-items [ string>number ] map ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 16:09:59 -05:00
										 |  |  | : fetch-mails ( uids data-spec -- texts )
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  |     [ comma-list ] dip "UID FETCH %s %s" sprintf "" command-response but-last ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 16:09:59 -05:00
										 |  |  | : copy-mails ( uids mailbox -- )
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  |     [ comma-list ] dip >utf7imap4 "UID COPY %s \"%s\"" sprintf "" | 
					
						
							|  |  |  |     command-response drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : append-mail ( mailbox flags date-time mail -- uid/f )
 | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         [ >utf7imap4 ] | 
					
						
							|  |  |  |         [ [ "" ] [ " " append ] if-empty ] | 
					
						
							|  |  |  |         [ timestamp>internal-date ] tri*
 | 
					
						
							|  |  |  |         "APPEND \"%s\" %s\"%s\"" sprintf | 
					
						
							|  |  |  |     ] dip utf8 encode command-response parse-append-mail ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-16 16:09:59 -05:00
										 |  |  | : store-mail ( uids command flags -- mail-flags )
 | 
					
						
							| 
									
										
										
										
											2014-01-16 14:20:36 -05:00
										 |  |  |     [ comma-list ] 2dip "UID STORE %s %s %s" sprintf "" command-response | 
					
						
							|  |  |  |     parse-store-mail ;
 | 
					
						
							| 
									
										
										
										
											2014-04-05 18:51:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | ! High level API | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : with-imap ( host email password quot -- )
 | 
					
						
							|  |  |  |     [ <imap4ssl> ] 3dip '[ _ _ login drop @ ] with-stream ; inline
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TUPLE: imap-settings host email password ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : <imap-settings> ( host email password -- obj )
 | 
					
						
							|  |  |  |     imap-settings new
 | 
					
						
							|  |  |  |         swap >>password | 
					
						
							|  |  |  |         swap >>email | 
					
						
							|  |  |  |         swap >>host ; inline
 | 
					
						
							| 
									
										
										
										
											2014-10-07 18:03:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-05 18:51:41 -04:00
										 |  |  | : with-imap-settings ( imap-settings quot -- )
 | 
					
						
							|  |  |  |     [ [ host>> ] [ email>> ] [ password>> ] tri ] dip with-imap ; inline
 |