factor/basis/smtp/server/server.factor

78 lines
2.0 KiB
Factor
Raw Normal View History

! Copyright (C) 2007 Elie CHAFTARI
! See http://factorcode.org/license.txt for BSD license.
2008-06-17 01:08:50 -04:00
USING: combinators kernel prettyprint io io.timeouts
2008-05-05 05:32:01 -04:00
sequences namespaces io.sockets continuations calendar
io.encodings.ascii io.streams.duplex destructors
locals concurrency.promises threads accessors ;
IN: smtp.server
! Mock SMTP server for testing purposes.
! $ telnet 127.0.0.1 4321
! Trying 127.0.0.1...
! Connected to localhost.
! Escape character is '^]'.
! 220 hello
! EHLO
! 220 and..?
! MAIL FROM: <here@mail.com>
! 220 OK
! RCPT TO: <there@mail.com>
! 220 OK
! Hi
! 500 ERROR
! DATA
! 354 Enter message, ending with "." on a line by itself
! Hello I am still waiting for your call
! Thanks
! .
! 220 OK
! QUIT
! bye
! Connection closed by foreign host.
SYMBOL: data-mode
: process ( -- )
readln {
{ [ [ dup "HELO" head? ] keep "EHLO" head? or ] [
"220 and..?\r\n" write flush t
] }
{ [ dup "QUIT" = ] [
"bye\r\n" write flush f
] }
{ [ dup "MAIL FROM:" head? ] [
"220 OK\r\n" write flush t
] }
{ [ dup "RCPT TO:" head? ] [
"220 OK\r\n" write flush t
] }
{ [ dup "DATA" = ] [
data-mode on
"354 Enter message, ending with \".\" on a line by itself\r\n"
write flush t
] }
{ [ dup "." = data-mode get and ] [
data-mode off
"220 OK\r\n" write flush t
] }
{ [ data-mode get ] [ dup global [ print ] bind t ] }
2008-04-11 13:57:43 -04:00
[
"500 ERROR\r\n" write flush t
2008-04-11 13:57:43 -04:00
]
} cond nip [ process ] when ;
:: mock-smtp-server ( promise -- )
#! Store the port we are running on in the promise.
[
"127.0.0.1" 0 <inet4> ascii <server> [
dup addr>> port>> promise fulfill
accept drop [
1 minutes timeouts
"220 hello\r\n" write flush
process
global [ flush ] bind
] with-stream
] with-disposal
] in-thread ;