2004-05-28 08:16:43 +02:00
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- SMTP client support for the Lua language.
|
|
|
|
-- LuaSocket toolkit.
|
|
|
|
-- Author: Diego Nehab
|
|
|
|
-- Conforming to RFC 2821
|
|
|
|
-- RCS ID: $Id$
|
|
|
|
-----------------------------------------------------------------------------
|
2004-05-28 09:47:41 +02:00
|
|
|
|
2004-06-04 17:15:45 +02:00
|
|
|
-----------------------------------------------------------------------------
|
2004-06-15 08:24:00 +02:00
|
|
|
-- Load required modules
|
2004-06-04 17:15:45 +02:00
|
|
|
-----------------------------------------------------------------------------
|
2004-06-15 08:24:00 +02:00
|
|
|
local smtp = requirelib("smtp")
|
2004-06-04 17:15:45 +02:00
|
|
|
local socket = require("socket")
|
|
|
|
local ltn12 = require("ltn12")
|
|
|
|
local tp = require("tp")
|
2004-05-28 08:16:43 +02:00
|
|
|
|
2004-06-04 17:15:45 +02:00
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- Setup namespace
|
|
|
|
-----------------------------------------------------------------------------
|
2004-06-15 08:24:00 +02:00
|
|
|
_LOADED["smtp"] = smtp
|
2003-10-21 03:12:23 +02:00
|
|
|
|
2004-06-15 08:24:00 +02:00
|
|
|
-- timeout for connection
|
|
|
|
TIMEOUT = 60
|
2004-03-19 06:04:03 +01:00
|
|
|
-- default server used to send e-mails
|
|
|
|
SERVER = "localhost"
|
2003-10-21 03:12:23 +02:00
|
|
|
-- default port
|
|
|
|
PORT = 25
|
2001-09-12 20:16:31 +02:00
|
|
|
-- domain used in HELO command and default sendmail
|
|
|
|
-- If we are under a CGI, try to get from environment
|
2003-10-21 03:12:23 +02:00
|
|
|
DOMAIN = os.getenv("SERVER_NAME") or "localhost"
|
2004-03-19 06:04:03 +01:00
|
|
|
-- default time zone (means we don't know)
|
|
|
|
ZONE = "-0000"
|
2003-10-21 03:12:23 +02:00
|
|
|
|
2004-03-22 05:15:03 +01:00
|
|
|
-- high level stuffing filter
|
|
|
|
function stuff()
|
|
|
|
return ltn12.filter.cycle(dot, 2)
|
|
|
|
end
|
|
|
|
|
2004-05-25 07:27:44 +02:00
|
|
|
---------------------------------------------------------------------------
|
|
|
|
-- Low level SMTP API
|
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
local metat = { __index = {} }
|
|
|
|
|
|
|
|
function metat.__index:greet(domain)
|
|
|
|
socket.try(self.tp:check("2.."))
|
|
|
|
socket.try(self.tp:command("EHLO", domain or DOMAIN))
|
|
|
|
return socket.try(self.tp:check("2.."))
|
|
|
|
end
|
|
|
|
|
|
|
|
function metat.__index:mail(from)
|
|
|
|
socket.try(self.tp:command("MAIL", "FROM:" .. from))
|
|
|
|
return socket.try(self.tp:check("2.."))
|
|
|
|
end
|
|
|
|
|
|
|
|
function metat.__index:rcpt(to)
|
|
|
|
socket.try(self.tp:command("RCPT", "TO:" .. to))
|
|
|
|
return socket.try(self.tp:check("2.."))
|
|
|
|
end
|
|
|
|
|
2004-05-26 06:58:32 +02:00
|
|
|
function metat.__index:data(src, step)
|
2004-05-25 07:27:44 +02:00
|
|
|
socket.try(self.tp:command("DATA"))
|
|
|
|
socket.try(self.tp:check("3.."))
|
2004-05-26 06:58:32 +02:00
|
|
|
socket.try(self.tp:source(src, step))
|
2004-05-25 07:27:44 +02:00
|
|
|
socket.try(self.tp:send("\r\n.\r\n"))
|
|
|
|
return socket.try(self.tp:check("2.."))
|
|
|
|
end
|
|
|
|
|
|
|
|
function metat.__index:quit()
|
|
|
|
socket.try(self.tp:command("QUIT"))
|
|
|
|
return socket.try(self.tp:check("2.."))
|
|
|
|
end
|
|
|
|
|
|
|
|
function metat.__index:close()
|
|
|
|
return socket.try(self.tp:close())
|
|
|
|
end
|
|
|
|
|
2004-03-19 06:04:03 +01:00
|
|
|
-- send message or throw an exception
|
2004-05-25 07:27:44 +02:00
|
|
|
function metat.__index:send(mailt)
|
|
|
|
self:mail(mailt.from)
|
2004-03-18 08:01:14 +01:00
|
|
|
if type(mailt.rcpt) == "table" then
|
|
|
|
for i,v in ipairs(mailt.rcpt) do
|
2004-05-25 07:27:44 +02:00
|
|
|
self:rcpt(v)
|
2004-03-18 08:01:14 +01:00
|
|
|
end
|
2003-10-21 03:12:23 +02:00
|
|
|
else
|
2004-05-25 07:27:44 +02:00
|
|
|
self:rcpt(mailt.rcpt)
|
2001-03-27 21:25:11 +02:00
|
|
|
end
|
2004-05-26 06:58:32 +02:00
|
|
|
self:data(ltn12.source.chain(mailt.source, stuff()), mailt.step)
|
2001-03-27 21:25:11 +02:00
|
|
|
end
|
|
|
|
|
2004-05-25 07:27:44 +02:00
|
|
|
function open(server, port)
|
2004-06-15 08:24:00 +02:00
|
|
|
local tp = socket.try(tp.connect(server or SERVER, port or PORT, TIMEOUT))
|
2004-05-25 07:27:44 +02:00
|
|
|
return setmetatable({tp = tp}, metat)
|
|
|
|
end
|
|
|
|
|
|
|
|
---------------------------------------------------------------------------
|
|
|
|
-- Multipart message source
|
|
|
|
-----------------------------------------------------------------------------
|
2004-03-19 06:04:03 +01:00
|
|
|
-- returns a hopefully unique mime boundary
|
2004-03-18 08:01:14 +01:00
|
|
|
local seqno = 0
|
|
|
|
local function newboundary()
|
|
|
|
seqno = seqno + 1
|
|
|
|
return string.format('%s%05d==%05u', os.date('%d%m%Y%H%M%S'),
|
|
|
|
math.random(0, 99999), seqno)
|
2001-09-12 20:16:31 +02:00
|
|
|
end
|
|
|
|
|
2004-03-22 05:15:03 +01:00
|
|
|
-- send_message forward declaration
|
|
|
|
local send_message
|
2004-03-19 06:04:03 +01:00
|
|
|
|
|
|
|
-- yield multipart message body from a multipart message table
|
2004-03-22 05:15:03 +01:00
|
|
|
local function send_multipart(mesgt)
|
2004-03-19 06:04:03 +01:00
|
|
|
local bd = newboundary()
|
|
|
|
-- define boundary and finish headers
|
|
|
|
coroutine.yield('content-type: multipart/mixed; boundary="' ..
|
|
|
|
bd .. '"\r\n\r\n')
|
|
|
|
-- send preamble
|
2004-06-15 08:24:00 +02:00
|
|
|
if mesgt.body.preamble then
|
|
|
|
coroutine.yield(mesgt.body.preamble)
|
|
|
|
coroutine.yield("\r\n")
|
|
|
|
end
|
2004-03-19 06:04:03 +01:00
|
|
|
-- send each part separated by a boundary
|
|
|
|
for i, m in ipairs(mesgt.body) do
|
|
|
|
coroutine.yield("\r\n--" .. bd .. "\r\n")
|
2004-03-22 05:15:03 +01:00
|
|
|
send_message(m)
|
2004-03-19 06:04:03 +01:00
|
|
|
end
|
|
|
|
-- send last boundary
|
|
|
|
coroutine.yield("\r\n--" .. bd .. "--\r\n\r\n")
|
|
|
|
-- send epilogue
|
2004-06-15 08:24:00 +02:00
|
|
|
if mesgt.body.epilogue then
|
|
|
|
coroutine.yield(mesgt.body.epilogue)
|
|
|
|
coroutine.yield("\r\n")
|
|
|
|
end
|
2004-03-19 06:04:03 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- yield message body from a source
|
2004-03-22 05:15:03 +01:00
|
|
|
local function send_source(mesgt)
|
2004-03-19 06:04:03 +01:00
|
|
|
-- set content-type if user didn't override
|
|
|
|
if not mesgt.headers or not mesgt.headers["content-type"] then
|
2004-03-19 07:14:56 +01:00
|
|
|
coroutine.yield('content-type: text/plain; charset="iso-8859-1"\r\n')
|
2004-03-19 06:04:03 +01:00
|
|
|
end
|
|
|
|
-- finish headers
|
|
|
|
coroutine.yield("\r\n")
|
|
|
|
-- send body from source
|
|
|
|
while true do
|
|
|
|
local chunk, err = mesgt.body()
|
|
|
|
if err then coroutine.yield(nil, err)
|
|
|
|
elseif chunk then coroutine.yield(chunk)
|
|
|
|
else break end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- yield message body from a string
|
2004-03-22 05:15:03 +01:00
|
|
|
local function send_string(mesgt)
|
2004-03-19 06:04:03 +01:00
|
|
|
-- set content-type if user didn't override
|
|
|
|
if not mesgt.headers or not mesgt.headers["content-type"] then
|
2004-03-19 07:14:56 +01:00
|
|
|
coroutine.yield('content-type: text/plain; charset="iso-8859-1"\r\n')
|
2004-03-19 06:04:03 +01:00
|
|
|
end
|
|
|
|
-- finish headers
|
|
|
|
coroutine.yield("\r\n")
|
|
|
|
-- send body from string
|
|
|
|
coroutine.yield(mesgt.body)
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
-- yield the headers one by one
|
2004-03-22 05:15:03 +01:00
|
|
|
local function send_headers(mesgt)
|
2004-03-18 08:01:14 +01:00
|
|
|
if mesgt.headers then
|
|
|
|
for i,v in pairs(mesgt.headers) do
|
|
|
|
coroutine.yield(i .. ':' .. v .. "\r\n")
|
|
|
|
end
|
|
|
|
end
|
2004-03-19 06:04:03 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- message source
|
2004-03-22 05:15:03 +01:00
|
|
|
function send_message(mesgt)
|
|
|
|
send_headers(mesgt)
|
|
|
|
if type(mesgt.body) == "table" then send_multipart(mesgt)
|
|
|
|
elseif type(mesgt.body) == "function" then send_source(mesgt)
|
|
|
|
else send_string(mesgt) end
|
2004-03-19 06:04:03 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
-- set defaul headers
|
2004-03-22 05:15:03 +01:00
|
|
|
local function adjust_headers(mesgt)
|
2004-06-04 17:15:45 +02:00
|
|
|
local lower = {}
|
2004-06-15 08:24:00 +02:00
|
|
|
for i,v in (mesgt.headers or lower) do
|
2004-06-04 17:15:45 +02:00
|
|
|
lower[string.lower(i)] = v
|
|
|
|
end
|
|
|
|
lower["date"] = lower["date"] or
|
2004-03-19 07:14:56 +01:00
|
|
|
os.date("!%a, %d %b %Y %H:%M:%S ") .. (mesgt.zone or ZONE)
|
2004-06-15 08:24:00 +02:00
|
|
|
lower["x-mailer"] = lower["x-mailer"] or socket.VERSION
|
2004-06-04 17:15:45 +02:00
|
|
|
-- this can't be overriden
|
|
|
|
lower["mime-version"] = "1.0"
|
|
|
|
mesgt.headers = lower
|
2001-03-27 21:25:11 +02:00
|
|
|
end
|
|
|
|
|
2004-03-18 08:01:14 +01:00
|
|
|
function message(mesgt)
|
2004-03-22 05:15:03 +01:00
|
|
|
adjust_headers(mesgt)
|
2004-03-19 06:04:03 +01:00
|
|
|
-- create and return message source
|
2004-03-22 05:15:03 +01:00
|
|
|
local co = coroutine.create(function() send_message(mesgt) end)
|
2004-06-15 08:24:00 +02:00
|
|
|
return function()
|
|
|
|
local ret, a, b = coroutine.resume(co)
|
|
|
|
if ret then return a, b
|
|
|
|
else return nil, a end
|
|
|
|
end
|
2001-03-27 21:25:11 +02:00
|
|
|
end
|
|
|
|
|
2004-05-25 07:27:44 +02:00
|
|
|
---------------------------------------------------------------------------
|
|
|
|
-- High level SMTP API
|
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
send = socket.protect(function(mailt)
|
2004-06-15 08:24:00 +02:00
|
|
|
local con = open(mailt.server, mailt.port)
|
|
|
|
con:greet(mailt.domain)
|
|
|
|
con:send(mailt)
|
|
|
|
con:quit()
|
|
|
|
return con:close()
|
2004-05-25 07:27:44 +02:00
|
|
|
end)
|
2001-03-27 21:25:11 +02:00
|
|
|
|
2003-10-21 03:12:23 +02:00
|
|
|
return smtp
|