邮件传输协议是一组规则,用于将邮件从源地址发送到目标地址,它可以控制邮件的传输模式,有需要的朋友们进入下文了解下Ruby实现发送邮件的两个方法吧!
Ruby通过smtp发邮件有2种操作手段,一是直接用Net::SMTP来发送,比较底层。如果还要发送附件,则需要额外gem:mailfactory,而mailfactory又依赖于包mime-types。即便如此如果邮件服务器连接需要ssl,则还需要require文件smtp-tls.rb,而这个rb文件需要openssl包的支持啊!第二种方法是直接使用高层的gem包mail,包含添加附件的功能,不过该gem也依赖于mime-types包,这个包专门用来描述邮件文件格式的,俗称多用途互联网邮件扩展类型啊。下面依次介绍下每种方法。
方法一:使用Net::SMTP
正常的非加密连接的smtp端口号为25,如果是加密则可能为587或465,要看具体的邮件服务器的说明。开始用的是QQ的邮件服务器,但老是不稳定,遂换为hotmail的,但hotmail需要ssl连接。按上面所述需要openssl包。麻烦开始了:rubygems.org用gem死活连不上,后来将openssl.gem直接下载到本地,用gem install -l openssl.gem安装,发现require还是出错,需要native ext:openssl.so动态库扩展,进入openssl的ext源代码目录用make编译提示出错:Ignore OpenSSL broken by Apple,提示我用其他的openssl库路径,我用毛啊!是不支持Mac OS X吗?但是检查了系统中是安装过openssl的:
动态库也是存在的,但不是ruby 的c_ext!
神马情况?考虑到tk对ruby的问题(见我另一篇在mac OS X下ruby使用tk的博文),我怀疑该ruby版本本身不支持原生openssl,这个版本是我在ruby-lang下载的ruby-2.1.5源代码编译并且安装的!遂用rvm下载了其ruby-2.1.5版本,一试竟然可以鸟!但随后发现hotmail.com的加密连接还是连不上,又换回QQ邮箱,用非加密的smtp,25端口连接。这回基本可以稳定发送了,如果换位163的邮箱测试发现更加稳定,上未重构的代码:
require 'net/smtp'
require './smtp-tls.rb'
require 'mailfactory'
#Senders and Recipients
from_name = 'localhost'
from_mail = '12345678@qq.com'
to_name = 'ks'
to_mail = '88888888@qq.com'
#Servers and Authentication
#smtp_host = 'smtp.qq.com'
smtp_host = 'smtp.163.com'
smtp_port = 25 #465 587 25
#smtp_domain = 'qq.com'
smtp_domain = 'localhost.localdomain'
smtp_user = "wangyi@163.com"
smtp_pwd = "xxxxxxxx"
#smtp_user = "12345678@qq.com"
#smtp_pwd = 'xxxxxxxx'
#The subject and the message
t = Time.now
subj = '1331 thinkpad test hopy'
msg_body = "send msg from ruby./n"
#The date/time should look something like: Thu, 03 Jan 2006 12:33:22 -0700
msg_date = t.strftime("%a, %d %b %Y %H:%M:%S +0800")
#Compose the message for the email
神马情况?考虑到tk对ruby的问题(见我另一篇在mac OS X下ruby使用tk的博文),我怀疑该ruby版本本身不支持原生openssl,这个版本是我在ruby-lang下载的ruby-2.1.5源代码编译并且安装的!遂用rvm下载了其ruby-2.1.5版本,一试竟然可以鸟!但随后发现hotmail.com的加密连接还是连不上,又换回QQ邮箱,用非加密的smtp,25端口连接。这回基本可以稳定发送了,如果换位163的邮箱测试发现更加稳定,上未重构的代码:
require 'net/smtp'
require './smtp-tls.rb'
require 'mailfactory'
#Senders and Recipients
from_name = 'localhost'
from_mail = '12345678@qq.com'
to_name = 'ks'
to_mail = '88888888@qq.com'
#Servers and Authentication
#smtp_host = 'smtp.qq.com'
smtp_host = 'smtp.163.com'
smtp_port = 25 #465 587 25
#smtp_domain = 'qq.com'
smtp_domain = 'localhost.localdomain'
smtp_user = "wangyi@163.com"
smtp_pwd = "xxxxxxxx"
#smtp_user = "12345678@qq.com"
#smtp_pwd = 'xxxxxxxx'
#The subject and the message
t = Time.now
subj = '1331 thinkpad test hopy'
msg_body = "send msg from ruby./n"
#The date/time should look something like: Thu, 03 Jan 2006 12:33:22 -0700
msg_date = t.strftime("%a, %d %b %Y %H:%M:%S +0800")
#Compose the message for the email
#{msg_body}
END_OF_MESSAGE
mail = MailFactory.new
mail.to = to_mail
mail.from = from_mail
mail.subject = subj
mail.text = msg_body
mail.attach(File.expand_path("./mail.rb")) #发送附件
#smtp = Net::SMTP.new(smtp_host,587)
#smtp.enable_starttls
#Net::SMTP.start(smtp_host, smtp_port, smtp_domain, smtp_user, smtp_pwd, :plain) do |smtp|
Net::SMTP.start(smtp_host,smtp_port,smtp_domain, smtp_user, smtp_pwd, :login) do |smtp|
#smtp.send_message msg, smtp_user, to_mail
#mail.to = to_mail
#puts smtp.methods
#smtp.enable_starttls
smtp.send_message(mail.to_s,smtp_user,to_mail)
end
方法二:使用ruby gem:mail(未完待续)
mail是一个比较高级的邮件库,包含了发送附件的功能。不过下载下来本地安装后开始报错提示refuse port 25之类的错误。开始我以为是权限不够,用sudo执行后,发现在执行require 'mail'命令时又报错!发现当初安装时未以sudo权限安装,先gem uninstall mail.gem,然后重新安装:sudo gem install mail.gem,此时在sudo和普通权限下require 'mail'都正常鸟:
require 'mail'
smtp = { :address => 'smtp.163.com', :port => 25, :domain => '163.com', /
:user_name => 'wangyi@163.com', :password => 'xxxxxxxx',/
:enable_starttls_auto => true, :openssl_verify_mode => 'none' }
Mail.defaults { delivery_method :smtp, smtp }
mail = Mail.new do
from 'wangyi@163.com'
to '12345678@qq.com'
subject 'test mail'
body 'body:hello send mail way 2 :)'
add_file File.expand_path("./mail2.rb")
end
mail.deliver!
有一个小问题,就是开始smtp中user_name和from里设置的发送邮件账户不一样,导致老是发送失败,这里都改成wangyi@163.com就可以了。但在方法一中可以不一样哦。还有2种方法中domain改成'localhost‘等其他值也能发送成功,好像没啥关系。
最后要说的是,2种方法都是用的非加密的方式连接,即邮件服务器允许非ssl方式连接,如果邮件服务器只能加密连接上面代码就不能用。至于这时该如何写代码,若知道的各位童鞋请不吝赐教哦。
最后,这两种方法都是以未加密的方式连接的,即邮件服务器允许非SSL连接,如果邮件服务器只能加密上面的连接代码,则不能使用。如果你了解这些知识,希望您给武林技术频道小编留言,大家相互学习,相互进步!
新闻热点
疑难解答
图片精选