258 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			258 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
|  | R"-++**++-(
 | ||
|  | ----------------------------------------------------------------------------- | ||
|  | -- SMTP client support for the Lua language. | ||
|  | -- LuaSocket toolkit. | ||
|  | -- Author: Diego Nehab | ||
|  | ----------------------------------------------------------------------------- | ||
|  | 
 | ||
|  | ----------------------------------------------------------------------------- | ||
|  | -- Declare module and import dependencies | ||
|  | ----------------------------------------------------------------------------- | ||
|  | local base = _G | ||
|  | local coroutine = require("coroutine") | ||
|  | local string = require("string") | ||
|  | local math = require("math") | ||
|  | local os = require("os") | ||
|  | local socket = require("socket") | ||
|  | local tp = require("socket.tp") | ||
|  | local ltn12 = require("ltn12") | ||
|  | local headers = require("socket.headers") | ||
|  | local mime = require("mime") | ||
|  | 
 | ||
|  | socket.smtp = {} | ||
|  | local _M = socket.smtp | ||
|  | 
 | ||
|  | ----------------------------------------------------------------------------- | ||
|  | -- Program constants | ||
|  | ----------------------------------------------------------------------------- | ||
|  | -- timeout for connection | ||
|  | _M.TIMEOUT = 60 | ||
|  | -- 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" | ||
|  | -- 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 | ||
|  | 
 | ||
|  | 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() | ||
|  |     return self.tp:close() | ||
|  | end | ||
|  | 
 | ||
|  | 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..")) | ||
|  | 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..")) | ||
|  | 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") | ||
|  |     end | ||
|  | end | ||
|  | 
 | ||
|  | -- send message or throw an exception | ||
|  | function metat.__index:send(mailt) | ||
|  |     self:mail(mailt.from) | ||
|  |     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 | ||
|  |     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)) | ||
|  |     local s = base.setmetatable({tp = tp}, metat) | ||
|  |     -- make sure tp is closed if we get an exception | ||
|  |     s.try = socket.newtry(function() | ||
|  |         s:close() | ||
|  |     end) | ||
|  |     return s | ||
|  | end | ||
|  | 
 | ||
|  | -- 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 | ||
|  | ----------------------------------------------------------------------------- | ||
|  | -- 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 | ||
|  | 
 | ||
|  | -- yield the headers all at once, it's faster | ||
|  | local function send_headers(tosend) | ||
|  |     local canonic = headers.canonic | ||
|  |     local h = "\r\n" | ||
|  |     for f,v in base.pairs(tosend) do | ||
|  |         h = (canonic[f] or f) .. ': ' .. v .. "\r\n" .. h | ||
|  |     end | ||
|  |     coroutine.yield(h) | ||
|  | end | ||
|  | 
 | ||
|  | -- yield multipart message body from a multipart message table | ||
|  | local function send_multipart(mesgt) | ||
|  |     -- make sure we have our boundary and send headers | ||
|  |     local bd = newboundary() | ||
|  |     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) | ||
|  |     -- send preamble | ||
|  |     if mesgt.body.preamble then | ||
|  |         coroutine.yield(mesgt.body.preamble) | ||
|  |         coroutine.yield("\r\n") | ||
|  |     end | ||
|  |     -- send each part separated by a boundary | ||
|  |     for i, m in base.ipairs(mesgt.body) do | ||
|  |         coroutine.yield("\r\n--" .. bd .. "\r\n") | ||
|  |         send_message(m) | ||
|  |     end | ||
|  |     -- send last boundary | ||
|  |     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 | ||
|  | end | ||
|  | 
 | ||
|  | -- yield message body from a source | ||
|  | local function send_source(mesgt) | ||
|  |     -- make sure we have a content-type | ||
|  |     local headers = lower_headers(mesgt.headers or {}) | ||
|  |     headers['content-type'] = headers['content-type'] or | ||
|  |         'text/plain; charset="iso-8859-1"' | ||
|  |     send_headers(headers) | ||
|  |     -- 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 | ||
|  | local function send_string(mesgt) | ||
|  |     -- make sure we have a content-type | ||
|  |     local headers = lower_headers(mesgt.headers or {}) | ||
|  |     headers['content-type'] = headers['content-type'] or | ||
|  |         'text/plain; charset="iso-8859-1"' | ||
|  |     send_headers(headers) | ||
|  |     -- send body from string | ||
|  |     coroutine.yield(mesgt.body) | ||
|  | end | ||
|  | 
 | ||
|  | -- message source | ||
|  | function send_message(mesgt) | ||
|  |     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 | ||
|  | end | ||
|  | 
 | ||
|  | -- set defaul headers | ||
|  | local function adjust_headers(mesgt) | ||
|  |     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) | ||
|  |     lower["x-mailer"] = lower["x-mailer"] or socket._VERSION | ||
|  |     -- this can't be overriden | ||
|  |     lower["mime-version"] = "1.0" | ||
|  |     return lower | ||
|  | end | ||
|  | 
 | ||
|  | function _M.message(mesgt) | ||
|  |     mesgt.headers = adjust_headers(mesgt) | ||
|  |     -- 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 | ||
|  | )-++**++-";
 |