luasocket/src/smtp.lua

256 lines
7.9 KiB
Lua
Raw Permalink Normal View History

2004-05-27 23:16:43 -07:00
-----------------------------------------------------------------------------
-- SMTP client support for the Lua language.
-- LuaSocket toolkit.
-- Author: Diego Nehab
-----------------------------------------------------------------------------
2004-05-28 00:47:41 -07:00
2004-06-04 08:15:45 -07:00
-----------------------------------------------------------------------------
-- Declare module and import dependencies
2004-06-04 08:15:45 -07:00
-----------------------------------------------------------------------------
2005-06-13 21:29:23 -07:00
local base = _G
2004-11-26 23:58:04 -08:00
local coroutine = require("coroutine")
local string = require("string")
local math = require("math")
local os = require("os")
2004-06-04 08:15:45 -07:00
local socket = require("socket")
local tp = require("socket.tp")
2004-06-04 08:15:45 -07:00
local ltn12 = require("ltn12")
2009-05-27 02:31:38 -07:00
local headers = require("socket.headers")
2004-06-15 21:28:21 -07:00
local mime = require("mime")
socket.smtp = {}
local _M = socket.smtp
-----------------------------------------------------------------------------
-- Program constants
-----------------------------------------------------------------------------
-- timeout for connection
_M.TIMEOUT = 60
2004-03-18 21:04:03 -08:00
-- default server used to send e-mails
_M.SERVER = "localhost"
-- default port
_M.PORT = 25
-- domain used in HELO command and default sendmail
-- If we are under a CGI, try to get from environment
_M.DOMAIN = os.getenv("SERVER_NAME") or "localhost"
2004-03-18 21:04:03 -08:00
-- default time zone (means we don't know)
_M.ZONE = "-0000"
---------------------------------------------------------------------------
-- Low level SMTP API
-----------------------------------------------------------------------------
local metat = { __index = {} }
function metat.__index:greet(domain)
self.try(self.tp:check("2.."))
self.try(self.tp:command("EHLO", domain or _M.DOMAIN))
return socket.skip(1, self.try(self.tp:check("2..")))
end
function metat.__index:mail(from)
self.try(self.tp:command("MAIL", "FROM:" .. from))
return self.try(self.tp:check("2.."))
end
function metat.__index:rcpt(to)
self.try(self.tp:command("RCPT", "TO:" .. to))
return self.try(self.tp:check("2.."))
end
2004-05-25 21:58:32 -07:00
function metat.__index:data(src, step)
self.try(self.tp:command("DATA"))
self.try(self.tp:check("3.."))
self.try(self.tp:source(src, step))
self.try(self.tp:send("\r\n.\r\n"))
return self.try(self.tp:check("2.."))
end
function metat.__index:quit()
self.try(self.tp:command("QUIT"))
return self.try(self.tp:check("2.."))
end
function metat.__index:close()
2004-06-18 14:41:44 -07:00
return self.tp:close()
end
2004-06-15 21:28:21 -07:00
function metat.__index:login(user, password)
self.try(self.tp:command("AUTH", "LOGIN"))
self.try(self.tp:check("3.."))
self.try(self.tp:send(mime.b64(user) .. "\r\n"))
self.try(self.tp:check("3.."))
self.try(self.tp:send(mime.b64(password) .. "\r\n"))
return self.try(self.tp:check("2.."))
2004-06-15 21:28:21 -07:00
end
function metat.__index:plain(user, password)
local auth = "PLAIN " .. mime.b64("\0" .. user .. "\0" .. password)
self.try(self.tp:command("AUTH", auth))
return self.try(self.tp:check("2.."))
2004-06-15 21:28:21 -07:00
end
function metat.__index:auth(user, password, ext)
if not user or not password then return 1 end
if string.find(ext, "AUTH[^\n]+LOGIN") then
return self:login(user, password)
elseif string.find(ext, "AUTH[^\n]+PLAIN") then
return self:plain(user, password)
else
self.try(nil, "authentication not supported")
2004-06-15 21:28:21 -07:00
end
end
2004-03-18 21:04:03 -08:00
-- send message or throw an exception
function metat.__index:send(mailt)
self:mail(mailt.from)
2004-11-26 23:58:04 -08:00
if base.type(mailt.rcpt) == "table" then
for i,v in base.ipairs(mailt.rcpt) do
self:rcpt(v)
end
else
self:rcpt(mailt.rcpt)
end
2004-06-16 23:23:13 -07:00
self:data(ltn12.source.chain(mailt.source, mime.stuff()), mailt.step)
end
function _M.open(server, port, create)
local tp = socket.try(tp.connect(server or _M.SERVER, port or _M.PORT,
_M.TIMEOUT, create))
2004-11-26 23:58:04 -08:00
local s = base.setmetatable({tp = tp}, metat)
-- make sure tp is closed if we get an exception
s.try = socket.newtry(function()
2004-06-21 21:49:57 -07:00
s:close()
end)
return s
end
2007-03-11 21:08:40 -07:00
-- convert headers to lowercase
local function lower_headers(headers)
local lower = {}
for i,v in base.pairs(headers or lower) do
lower[string.lower(i)] = v
end
return lower
end
---------------------------------------------------------------------------
-- Multipart message source
-----------------------------------------------------------------------------
2004-03-18 21:04:03 -08:00
-- returns a hopefully unique mime boundary
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)
end
-- send_message forward declaration
local send_message
2004-03-18 21:04:03 -08:00
-- yield the headers all at once, it's faster
2009-05-27 02:31:38 -07:00
local function send_headers(tosend)
local canonic = headers.canonic
local h = "\r\n"
2009-05-27 02:31:38 -07:00
for f,v in base.pairs(tosend) do
h = (canonic[f] or f) .. ': ' .. v .. "\r\n" .. h
end
coroutine.yield(h)
end
2004-03-18 21:04:03 -08:00
-- yield multipart message body from a multipart message table
local function send_multipart(mesgt)
-- make sure we have our boundary and send headers
2004-03-18 21:04:03 -08:00
local bd = newboundary()
2007-03-11 21:08:40 -07:00
local headers = lower_headers(mesgt.headers or {})
headers['content-type'] = headers['content-type'] or 'multipart/mixed'
headers['content-type'] = headers['content-type'] ..
'; boundary="' .. bd .. '"'
send_headers(headers)
2004-03-18 21:04:03 -08:00
-- send preamble
if mesgt.body.preamble then
coroutine.yield(mesgt.body.preamble)
coroutine.yield("\r\n")
end
2004-03-18 21:04:03 -08:00
-- send each part separated by a boundary
2004-11-26 23:58:04 -08:00
for i, m in base.ipairs(mesgt.body) do
2004-03-18 21:04:03 -08:00
coroutine.yield("\r\n--" .. bd .. "\r\n")
send_message(m)
2004-03-18 21:04:03 -08:00
end
-- send last boundary
2004-03-18 21:04:03 -08:00
coroutine.yield("\r\n--" .. bd .. "--\r\n\r\n")
-- send epilogue
if mesgt.body.epilogue then
coroutine.yield(mesgt.body.epilogue)
coroutine.yield("\r\n")
end
2004-03-18 21:04:03 -08:00
end
-- yield message body from a source
local function send_source(mesgt)
-- make sure we have a content-type
2007-03-11 21:08:40 -07:00
local headers = lower_headers(mesgt.headers or {})
headers['content-type'] = headers['content-type'] or
'text/plain; charset="iso-8859-1"'
send_headers(headers)
2004-03-18 21:04:03 -08:00
-- send body from source
while true do
2004-03-18 21:04:03 -08:00
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
local function send_string(mesgt)
-- make sure we have a content-type
2007-03-11 21:08:40 -07:00
local headers = lower_headers(mesgt.headers or {})
headers['content-type'] = headers['content-type'] or
'text/plain; charset="iso-8859-1"'
send_headers(headers)
2004-03-18 21:04:03 -08:00
-- send body from string
coroutine.yield(mesgt.body)
end
-- message source
function send_message(mesgt)
2004-11-26 23:58:04 -08:00
if base.type(mesgt.body) == "table" then send_multipart(mesgt)
elseif base.type(mesgt.body) == "function" then send_source(mesgt)
else send_string(mesgt) end
2004-03-18 21:04:03 -08:00
end
-- set defaul headers
local function adjust_headers(mesgt)
2007-03-11 21:08:40 -07:00
local lower = lower_headers(mesgt.headers)
lower["date"] = lower["date"] or
os.date("!%a, %d %b %Y %H:%M:%S ") .. (mesgt.zone or _M.ZONE)
2005-06-13 21:29:23 -07:00
lower["x-mailer"] = lower["x-mailer"] or socket._VERSION
2004-06-04 08:15:45 -07:00
-- this can't be overriden
lower["mime-version"] = "1.0"
2007-03-11 21:08:40 -07:00
return lower
end
function _M.message(mesgt)
2007-03-11 21:08:40 -07:00
mesgt.headers = adjust_headers(mesgt)
2004-03-18 21:04:03 -08:00
-- create and return message source
local co = coroutine.create(function() send_message(mesgt) end)
return function()
local ret, a, b = coroutine.resume(co)
if ret then return a, b
else return nil, a end
end
end
---------------------------------------------------------------------------
-- High level SMTP API
-----------------------------------------------------------------------------
_M.send = socket.protect(function(mailt)
local s = _M.open(mailt.server, mailt.port, mailt.create)
local ext = s:greet(mailt.domain)
s:auth(mailt.user, mailt.password, ext)
s:send(mailt)
s:quit()
return s:close()
end)
return _M