我想使用Rust获取我的GMail电子邮件。
我目前正在使用rust-openssl库和标准库中的TcpStream
类。
我遇到的问题很简单:以下代码在println!("0")
方法中println!("1")
和login
之间“卡住”。溪流突然“冻结”。
尽管如此,它在向另一个URL执行HTTP请求时效果很好。
extern crate openssl;
use std::io::{IoResult, TcpStream};
use openssl::ssl::{SslContext, Sslv23, SslStream, SslVerifyPeer};
pub struct GmailSocket {
_stream: SslStream<TcpStream>
}
impl GmailSocket {
pub fn connect() -> IoResult<GmailSocket> {
let tcpstream = try!(TcpStream::connect("imap.gmail.com", 993));
let mut sslcontext = SslContext::new(Sslv23);
match sslcontext.set_CA_file("./cert.pem") {
None => {},
Some(e) => fail!("{}", e)
}
sslcontext.set_verify(SslVerifyPeer, None);
let sslstream = SslStream::new(&sslcontext, tcpstream);
Ok(GmailSocket {_stream: sslstream})
}
pub fn login(&mut self, username: &str, password: &str) -> IoResult<bool> {
let req = format!("L01 LOGIN {:s} {:s}\r\n", username, password);
try!(self._stream.write_str(req.as_slice()));
println!("0");
match try!(self._stream.read_to_string()) {
res => println!("{}", res)
}
println!("1");
Ok(true)
}
}
fn main() {
let mut gmail_socket = match GmailSocket::connect() {
Ok(s) => s,
Err(e) => fail!("{}", e)
};
gmail_socket.login("account@gmail.com", "password");
}
问题是来自SSL认证文件,IMAP请求的格式,代码还是自己使用的方法? OAuth是否必须执行此类任务?
此外,使用此命令生成了./cert.pem
文件:
openssl req -x509 -nodes -days 365 \
-newkey rsa:1024 -keyout cert.pem -out cert.pem
由于本练习的主要目的是教育性的,我想避免使用rust-http。
答案 0 :(得分:1)
我会调查imap crate。
我认为您的问题来自IMAP请求的格式,该格式遵循特定协议。
使用imap crate,您可以轻松地通过SSL连接到收件箱,如下所示:
fn connect_to_email() -> IMAPStream {
match IMAPStream::connect("imap.gmail.com",
993,
Some(SslContext::new(SslMethod::Sslv23).unwrap())) {
Ok(s) => s,
Err(e) => panic!("connect to email {}", e),
}
}
然后,要实际抓取电子邮件,您可以选择一个文件夹,然后获取相应的文本:
imap_socket.select("INBOX")
imap_socket.fetch(str_num, "BODY[TEXT]")
imap_socket.fetch(str_num, "BODY[HEADER]")
其中imap_socket : &mut IMAPStream
和str_num : &str
(收件箱中的电子邮件索引,按时间戳排序)。