概述
当你想知道自己发送的邮件是否被收信人查看过时,使用电子邮件追踪系统(email tracking system)就可以帮助你。
使用方法
打开链接电子邮件追踪系统,输入email地址和标题,选择接收通知的次数,然后激活获取tracking image,在一分钟内将其复制到你的email邮件正文中,再正常发送邮件就行了。
实现原理
由于标签的src属性会主动引入外部文件,所以将调用“tracker程序”(此程序会正常输出一张图片)的url作为src的值,并将此放入邮件正文中与之一起发送出去。这样,每当收信人打开该邮件显示该时,都会调用“tracker程序”,这时“tracker程序”会发送email通知你。而邮件也必须是html格式的才行。
程序说明
程序共有四部分:
index.html -- 创建一个email tracker的程序界面,需要传递三个参数 - 邮件地址,标题和接收通知的次数。tracker.php -- 接收参数产生一个email tracker。blank.php -- 发送email通知用户邮件已阅,并生成一个图片。msg_template.html -- 通知正文的模板。 代码
创建表的sql:
email_tracker sql
create table `email_tracker` (
`unique_id` varchar(50) not null,
`email` varchar(100) default null,
`title` varchar(100) default null,
`number` tinyint(4) default '3',
`times` tinyint(4) default '-1',
`ip` varchar(16) default null,
`sent_time` int(11) default '0',
primary key (`unique_id`),
unique key `unique_id` (`unique_id`)
) engine=myisam;
tracker.php接收参数产生一个新的email tracker:
tracker.php
$db = get_db();
$ip = $_server['remote_addr']?$_server['remote_addr']:'127.0.0.1'; //获取用户ip
$unique_id = get_unique_id($ip); // 产生一个唯一id
$number = intval($_post['number']);
$email = trim($_post['email']);
$title = trim($_post['title']);
$sent_time = time(); // 作为发送邮件的时间
$db->query(insert into `email_tracker` (unique_id, email, title, number, ip, sent_time) values ('$unique_id', '$email', '$title', $number, '$ip', '$sent_time'));
用于产生唯一id的get_unique_id函数:
get_unique_id
function current_microsecond() {
list($usec, $sec) = explode( ,microtime());
return $sec.substr($usec, 2);
}
// 获取随机数
function random() {
$tmp = rand(0,1)?'-':'';
return $tmp.rand(1000, 9999).rand(1000, 9999).rand(1000, 9999).rand(100, 999).rand(100, 999);
}
// 产生一个伪guid
// 三段 : 一段是地址 一段是微秒 一段是随机数
function get_unique_id($ip) {
$raw = strtoupper(md5($ip.'-'.current_microsecond().'-'.random()));
return substr($raw,0,8).'-'.substr($raw,8,4).'-'.substr($raw,12,4).'-'.substr($raw,16,4).'-'.substr($raw,20);
}
blank.php由邮件正文中tracking image调用,发送email通知用户邮件已阅,并生成一个图片。
blank.php
if(!($unique_id = trim($_server['query_string']))) exit_with_image_blank();
$db = get_db();
$tracker = $db->fetch_one(select * from `email_tracker` where unique_id='$unique_id');
// 记录不存在,或tracking已经结束
if(empty($tracker) && $tracker['times'] >= $tracker['number']) {
// 输出一个空白图片并退出
exit_with_image_blank();
}
// 邮件发送时到现在经过的时间
$time_elapsed = time() - $tracker['sent_time'];
// 不到一分钟
if($time_elapsed 60) {
if($tracker['times'] 0) { // 还未激活email tracker
$db->query(update `email_tracker` set times=0 where unique_id='$unique_id');
}
exit_with_image_blank();
}
// 一分钟之后,times
if($tracker['times'] 0) {
$one_minute_ago = time() - 60;
// 删除所有经过一分钟还未激活的email tracker
$db->query(delete from `email_tracker` where times $one_minute_ago);//unique_id='$unique_id'
exit_with_image_blank();
}
// 获取收信人ip
$rcpt_ip = $_server['remote_addr']?$_server['remote_addr']:'127.0.0.1';
if($rcpt_ip == $tracker['ip']) {
//与用户ip相同,也许是用户自己打开了email
$tracker['times']++;
} else {
// 是收信人打开了email, 查阅次数增加一次
$tracker['times']++;
}
$db->query(update `email_tracker` set times=$tracker[times] where unique_id='$unique_id');
if($tracker['times'] >= $tracker['number']) {
// tracking已经结束, 删除记录
$db->query(delete from `email_tracker` where unique_id='$unique_id');
}
// 发送email
send_mail('mailtracker0@gmail.com', $tracker['email'], array(
'subject' =>'your email '.$tracker['title'].' has been read!',
'body' =>notify_content($tracker, $rcpt_ip, $time_elapsed),
'headers' =>mime-version: 1.0;\r\ncontent-type:text/html; charset=\utf-8\;\r\n,
'host' =>'smtp.gmail.com',
'ssl' =>true,
'port' =>465,
'auth' =>true,
'user' =>'mailtracker0',
'pass' =>'**********'
));
// 输出一个空白图片并退出
exit_with_image_blank();
send_mail函数用于发送通知邮件:
send_mail
/*
* 参数:
* from required 发信人email
* recipients required 收信人email地址,如果有多个email则以','分隔或传递数组
* params optional 其它可选参数组成的数组
* ----subject 邮件主题
* ----body 邮件正文
* ----headers
* ----host 用于发邮件的smtp主机
* ----port 端口
* ----timeout 超时时间
* ----ssl 是ssl加密,默认为false
* ----auth 是否要身份验证,默认为false
* ----user 用于身份验证的用户名
* ----pass 用于身份验证的密码
**/
function send_mail($from, $recipients, $params = array()) {
if(empty($from) || empty($recipients) || !is_array($params)) return 'params error.';
define('crlf', \r\n);
$port = 25;
$host = 'localhost';
$timeout = 10;
$auth = false;
$ssl = false;
$subject = 'untitled';
foreach ($params as $key => $value) {
$$key = $value;
}
if(!is_array($recipients)) $recipients = explode(',', trim($recipients));
if(!is_array($headers)) $headers = explode(crlf, trim($headers));
if(!is_string($body)) $body = '';
$body = str_replace(crlf . '.', crlf . '..', $body{0} == '.' ? '.'.$body : $body);
// 连接smtp服务器
$connection = fsockopen($ssl?'ssl://'.$host:$host, $port, $errno, $errstr, $timeout);
if(!is_resource($connection)) {
return 'failed to connect to server: '.$errstr;
}
while($line = @fgets($connection, 1024)) if($line{3} == ' ') break;
// 保存命令
$datas = array();
if($auth === true) { // 需要验证身份
$datas[] = array('ehlo '.$host.crlf, '250', 'ehlo command failed, output: ');
$datas[] = array('auth login'.crlf , '334', 'auth command failed, output: ');
$datas[] = array(base64_encode($user).crlf, '334', 'auth command failed, output: ');
$datas[] = array(base64_encode($pass).crlf, '235', 'auth command failed, output: ');
} else {
$datas[] = array('helo '.$host.crlf, '250', 'helo command failed, output: ');
}
// 设置发信人
$datas[] = array('mail from: '.$from.'>'.crlf, '250', 'mail from error, output: ');
// 设置收信人
foreach($recipients as $value) {
$datas[] = array('rcpt to: '.$value.'>'.crlf, '250', 'rcpt to error, output: ');
}
$datas[] = array('data'.crlf, '354', 'data command failed, output: ');
// 邮件headers
$datas[] = 'from: '.$from.crlf;
$datas[] = 'subject: '.$subject.crlf;
foreach($headers as $value) {
$datas[] = $value
