免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
123下一页
最近访问板块 发新帖
查看: 24025 | 回复: 23
打印 上一主题 下一主题

自己做的短信机器人(windows版) [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-05-21 01:40 |只看该作者 |倒序浏览
一、运行程序的一些准备。
1.下载并安装需要的模块
device::modem     device::gsm        Win32-SerialPort
这三个模块在cpan上都有,先下好安装,其中GSM.pm 该模块的_send_sms_pdu方法被修改过,原模块不支持中文短信,修改过后的模块支持中文短信。CommPort.pm 该模块的第791行被修改过原$self->{"_L_BAUD"} = {};  修改后为$self->{"_L_BAUD"} = {19200 => 19200};原因是未修改时运行出错找不到BAUD值,因此强制设定为19200。对sms.pm进行修改,sms.pm的作用是在串口接收到短信的信息时,对每条短信进行解码还原,但是原模块是针对特定的设计的,因此只对 AT+CMGL=4即提取所有短信这条命令能识别,对对70的判断语句多了对AT+CMGR命令的支持,前一命令返回参数有4个,后一命令返回参数有3个,为了对应,将后一个命令的第一参数匹配了空!

2.修改后的三个模块我上传到这里,下载后覆盖相应的文件即可。

   修改后的三个模块.rar (33.37 KB, 下载次数: 308)

3.准备一个支持蓝牙的手机,并且手机要能够使用AT命令。(测试手机是否支持AT命令的方法,windows下用超级终端,设置好发送几个AT命令看返回就可以了,nokia手机6300只支持发送,其他好像都不支持,最后找了个天语的,什么命令都支持,山寨机的好处)

这些准备好基本上可以运行程序了。下面发代码:
  1. # Gsmbot - a perl program for sending and parsing SMS
  2. #          main function is to send sms to group and in certain
  3. #          time statistc answer which you received,
  4. # programme by Y.K
  5. # Email:yk_redrum@163.com

  6. # 待删
  7. # 目前需要实现的功能
  8. # 1.同一个人要是回复多次,程序要重复统计
  9. # 2.群发的时候,发短息给手机,手机不接收
  10. # 3.定时清除手机内的短信
  11. # 4.待发现

  12. use strict;
  13. use Device::Gsm;
  14. use Encode;
  15. use Time::Local;

  16. #
  17. # master tel number sending command
  18. #
  19. my $masterTelnum = 134xxxxxxx7;
  20. #
  21. #group book
  22. #
  23. our %team = (

  24.       1588xxxxx17 => '张三',
  25.       1343xxxxx37 => '李四,
  26. );

  27. #
  28. # all name of all phone book
  29. #
  30. my %pbName =(team=>undef);
  31. #
  32. # chr(10) eq \n in sms
  33. #
  34. my $n = chr(10);
  35. #
  36. # seventy unicode charactor of a sms
  37. #
  38. my $gsmStatus ="请输入命令,格式:$n<别名>#<时间>#<内容>$n别名:只支持team。$n时间:单位是分钟,格式为数字。$n内容:内容不要包含符号#。";

  39. my $newMeasageReg = qr/^\+CMTI:\s*\"\w*\",\s*\d+$/m;

  40. my($startTime,$interval) = (undef,undef);;

  41. my $pin  = '0000';

  42. #
  43. # main procedure
  44. #

  45. my $gsm = new Device::Gsm( port => 'COM4', pin => $pin,log => 'file,send.log' );    # loglevel => 'debug'

  46. die "cannot create Device::Gsm object!" unless $gsm;

  47. $gsm->connect( baudrate => 19200 ) or die "cannot connect to GSM device on [COM4\]\n";

  48. $gsm->register() or die "cannot register on GSM network: check pin and/or network signal!";

  49. print "\n[initial]:\tok! connected and registered to network.\n";

  50. print "[initial]:\tSet CNMI to [2,1,0,0,0]...  \n";
  51. $gsm->atsend( qq[AT+CNMI=2,1,0,0,0] . Device::Modem::CR );
  52. $gsm->answer($Device::Modem::STD_RESPONSE);

  53. print "[initial]:\tSet CPMS to [MT]...  \n";
  54. $gsm->atsend( qq[AT+CPMS="MT"] . Device::Modem::CR );
  55. $gsm->answer($Device::Modem::STD_RESPONSE);

  56. print "[initial]:\tWaiting for command!...\n\n";

  57. while(1){
  58.         #
  59.         # statistic
  60.         #
  61.         if(defined $startTime && defined $interval){
  62.              my $now = time;
  63.              my $endTime = $startTime + $interval;
  64.              print "[timer]:\t$now ->($startTime , $endTime)\n";
  65.              if( $now > $endTime){ #time up

  66.                 print "[statistic]:\tstatistic start.....\n";

  67.                 statistic($gsm,$startTime,$endTime);

  68.                 #destroy timer
  69.                ($startTime,$interval) = (undef,undef);

  70.                print "[statistic]:\tstatistic over and timer has been destroied!\n";
  71.              }
  72.         }



  73.         #read until a new message recieve with master command in it
  74.         next unless (my $haveNewMeasage = $gsm->answer($newMeasageReg) =~ m/\+CMTI:\s*\"\w*\",\s*(\d+)/ );

  75.         #
  76.         #read one sms according to +CMTI:......
  77.         #

  78.         print "[new sms]:\tat position \[$1\] A new sms has been received,and reading ...\n";
  79.         my $msg = readSMS($gsm,$1);

  80.         if( ref $msg ) {

  81.                 print "[new sms]:\tparsing command ...\n";

  82.                 if($msg->sender =~ /$masterTelnum/o){
  83.                     my $smsContent = encode( "gb2312", $msg->content() );

  84.                     #remove all \n \s and blank
  85.                     $smsContent =~ s/\s//g;

  86.                     #
  87.                     #parse command
  88.                     #

  89.                     if($smsContent =~ /\#/){ # phone_book#interval_time#sms_content

  90.                         print "[command parse]:\tcommand is group send ...\n";

  91.                         #group send
  92.                         my($gpName,$gpTime,$gpMsg)= split(/\#/,$smsContent);

  93.                         $gpName =lc $gpName;

  94.                         if(exists $pbName{$gpName} && $gpTime =~ /^\d+$/){

  95.                            print "[group send]:\tparametre set:groupName:$gpName\tinterval: $gpTime(min)\n";
  96.                            print "[group send]:\tnow group sending ...\n";

  97.                            my($sendlog,$rgpName);
  98.                            {
  99.                              #allow Symbolic references
  100.                              no strict 'refs';
  101.                              $rgpName = \%{$gpName};
  102.                              #send group sms,and return result
  103.                              $sendlog = sendGp($gsm,$rgpName,$gpMsg);
  104.                            }

  105.                            #send accomplished ,now send log to master!
  106.                            if($sendlog){

  107.                               #
  108.                               #format report
  109.                               #
  110.                               groupReport($gsm,\$sendlog,$rgpName);

  111.                               #
  112.                               #timer start after $interval statistic sms
  113.                               #
  114.                               print "[timer]:\tcreative timer\n";

  115.                               $startTime = time;          #begin count
  116.                               $interval = $gpTime * 60;    #convert to secs

  117.                            }else{
  118.                               print "[group send]:\tlack sms content which you want send!\n";
  119.                            }
  120.                         }else{
  121.                               print "[group send]:\tfirst and second parametre format of command is wrong or empty!\n";
  122.                         }

  123.                     }elsif( lc($smsContent) =~ /status/ ){
  124.                            #single line command

  125.                            print "[command parse]:\tcommand is status\n";

  126.                            print "[command status]:\tsmsBot status sending ...\n";

  127.                            my $lOk = $gsm->send_sms(
  128.                                      content => $gsmStatus,
  129.                                      recipient => $masterTelnum,
  130.                                      encode=>'chinese',
  131.                                      status_report=>1,
  132.                               );
  133.                            if( $lOk ) {
  134.                                   #
  135.                                   #send ok to log
  136.                                   #
  137.                                   print "[command status]:\t--smsBot status is sent successful\n";
  138.                            } else {
  139.                                   #
  140.                                   #send error to log
  141.                                   #
  142.                                   print "[command status]:\t--smsBot status is sent successful\n";
  143.                            }

  144.                     }else{

  145.                           print "[command parse]:\tcommon message and no command\n";

  146.                     }

  147.                 }

  148.         } else {
  149.               #
  150.               #log
  151.               #
  152.               print "[new sms]:\tsms at position $1 read failed!\n";
  153.         }

  154. }

  155. #
  156. # sendGp
  157. #
  158. # send SMS to every one of group phone book...
  159. # hashref = sendGp($obj,$r_phonebook_hash,$gpcontent)
  160. #

  161. sub sendGp {

  162.   my $self = shift;
  163.   my( $bookref,$gpMsg )  = @_;
  164.   my $log = {};

  165.   return undef unless $gpMsg;

  166.   foreach (keys(%$bookref)){

  167.     print "[group send]:\tsending sms to $bookref->{$_} ...\n";

  168.     my $lOk = $self->send_sms(
  169.                     content => $gpMsg,
  170.                     recipient => $_,
  171.                     encode=>'chinese',
  172.                     status_report=>1,
  173.                     );
  174.     if( $lOk ) {
  175.         # send successful
  176.         print "[group send]:\t--sms to $bookref->{$_} has been sent successful\n";
  177.         $log->{$_} = 1;   #send ok
  178.      }else {
  179.         #send error to log
  180.         print "[group send]:\t--sms to $bookref->{$_} has been sent fail\n";
  181.         $log->{$_} = 0;   #send fail
  182.      }
  183.      #select(undef,undef,undef,0.5);
  184.   }
  185.   return $log;
  186. }

  187. #
  188. # readSMS
  189. #
  190. # read single readSMS
  191. # $msgobj = readSMS($obj,position)
  192. #

  193. sub readSMS {

  194.   my( $self,$pos ) = @_;
  195.   return undef unless $pos;

  196.   #
  197.   # when invoke send,procedure set ATE1,and maybe first pair of @data is not data of message!
  198.   # so set it off ATE0

  199.   $self->atsend(  q[ATE0] . Device::Modem::CR );
  200.   $self->answer($Device::Modem::STD_RESPONSE);

  201.   $self->atsend( qq[AT+CMGR=$pos] . Device::Modem::CR );
  202.   my $message =$self->answer($Device::Modem::STD_RESPONSE);

  203.   # Catch the case that response you wanted have not received because time is up.
  204.   while (my $more = $self->answer($Device::Modem::STD_RESPONSE, 200)) {
  205.   #-- $self->answer will chomp trailing newline, add it back
  206.       $message .= $more;
  207.   }

  208. my @data = split /[\r\n]+/m, $message;


  209.   # Current sms storage memory (ME or SM)
  210.   my $storage = $self->storage();
  211.   #
  212.   # Parse received data (result of +CMGR command)
  213.   #

  214.   $self->log->write('debug', 'data[] = ', $data[0] );

  215.   my $header=shift @data;
  216.   my $pdu   =shift @data;
  217.   # Instance new message object
  218.   my $msg = new Device::Gsm::Sms(
  219.                 header => $header,
  220.                 pdu    => $pdu,
  221.                 storage => $storage,
  222.                 parent => $gsm                  # Ref to parent Device::Gsm class
  223.   );
  224.   return $msg;
  225. }

  226. #
  227. # statistic
  228. #
  229. # statistic answers of team
  230. # $sok = statistic($obj,$startTime,$endTime)
  231. #

  232. sub statistic {

  233.     my($self,$startTime,$endTime) = @_;

  234.     #shut up echo
  235.     $self->atsend(  q[ATE0] . Device::Modem::CR );
  236.     $self->answer($Device::Modem::STD_RESPONSE);

  237.     my @sms = $self->messages;

  238.     if( @sms ) {

  239.          #report info
  240.          my $report = {ok=>0 , no=>0};
  241.          my $num = 0;
  242.          my $satisticTime = $interval/60;

  243.          foreach( @sms ) {

  244.              my $sender = $_->sender();
  245.              my @tdata = split /\s/,$_->time();
  246.              my ($mday,$mon,$year) = split /\//,$tdata[0];
  247.              my ($hour,$min,$sec) = split /:/,$tdata[1];
  248.              $mon  -= 1;
  249.              $year += 100;
  250.              my $smsTime = timelocal($sec,$min,$hour,$mday,$mon,$year);

  251.              if(exists $team{substr($sender,3)}){

  252.                  my $csender = $team{substr($sender,3)};

  253.                  print "[statistic]:\tsender:$csender\[$sender\]\ttime:$smsTime($startTime,$endTime)\n";

  254.                  if( $smsTime >= $startTime && $smsTime <= $endTime ){

  255.                      $num++;
  256.                      my $content = encode( "gb2312", $_->content() );

  257.                      print "[statistic]:\t--matched:\n";
  258.                      print "            \t--sender:$csender\[$sender\] time:$smsTime($startTime,$endTime)\n";
  259.                      print "            \t--content:$content\n";

  260.                      $content =~ s/\s//g;
  261.                      ($content =~ /好|要的|我要来|算上我|yes|恩|啊/ ) ? $report->{ok}++ : $report->{no}++;

  262.                  }else{
  263.                       print "[statistic]:\t--time of sms is not fit\n";
  264.                  }
  265.              }else{
  266.                   print "[statistic]:\tsender:$sender do not belong to %team!\n";
  267.              }
  268.          }

  269.          print "[statistic]:\tstatistic report sending ...\n";

  270.          my $reportContent = "统计时间:$satisticTime分钟,一共$num人回复了短信,其中同意的有$report->{ok}人,不同意的有$report->{no}人";
  271.          my $lOk = $gsm->send_sms(
  272.                    content =>$reportContent,
  273.                    recipient =>$masterTelnum,
  274.                    encode=>'chinese',
  275.                    status_report=>1,
  276.                       );
  277.          if( $lOk ) {
  278.                     ##
  279.                     #send ok  to log
  280.                     #
  281.                     print "[statistic]:\tstatistic report has been sent successful\n";
  282.          } else {
  283.                     ##
  284.                     #send error to log
  285.                     #
  286.                     print "[statistic]:\tstatistic report has been sent fail\n";
  287.          }
  288.     }
  289. }

  290. #
  291. # groupReport
  292. #
  293. # format result of group send
  294. # groupReport($obj,$r_sendlog,$r_phonebook_hash)
  295. #

  296. sub groupReport {

  297.     my($self,$rsendlog,$rbook) = @_;

  298.     my ($nok,$nerror)=(0,0);
  299.     my $tReport;

  300.     while( my($key, $value) = each %$$rsendlog) {

  301.            if($value){
  302.               #send ok
  303.               $nok++
  304.            }else{
  305.               #send fail
  306.               $tReport .= "$rbook->{$key},";
  307.               $nerror++

  308.            }
  309.            #($value) ? $nok++ : $nerror++;

  310.     }

  311.     my $report = "短信群发报告$n发送成功$nok个,发送失败$nerror个$n";

  312.     if($nerror){

  313.        $report .="发送失败包括:";
  314.        $report .= $tReport;

  315.     }

  316.     #length of sms content:a new involve 70 charactor
  317.     my $ucreport = decode("gb2312",$report);
  318.     my $wlen = length($ucreport);
  319.     if($wlen > 70){

  320.        print "[group send]:\tnow sending mutipage report ...\n";

  321.        #67 =  70 - lenghth("2/3") --"2/3" is page of sms
  322.        my $pages = int($wlen/67);
  323.        $pages++ if $wlen%67;

  324.        foreach(0..$pages-1){

  325.           my $reportpage;
  326.           my $cpage = $_+1;
  327.           $reportpage  = encode("gb2312",substr($ucreport,$_*67,67));
  328.           $reportpage .= "$cpage/$pages";

  329.           my $lOk = $gsm->send_sms(
  330.                           content => $reportpage,
  331.                           recipient => $masterTelnum,
  332.                           encode=>'chinese',
  333.                           status_report=>1,
  334.                           );
  335.           if( $lOk ) {
  336.                  #
  337.                  #send ok to log
  338.                  #
  339.                  print "[group send]:\t--page $cpage of report is sent successful\n";
  340.           } else {
  341.                  #
  342.                  #send error to log
  343.                  #
  344.                  print "[group send]:\t--page $cpage of report is sent fail\n";
  345.           }
  346.        }

  347.     }else{

  348.          print "[group send]:\tnow sending single page report ...\n";
  349.          my $lOk = $gsm->send_sms(
  350.                    content => $report,
  351.                    recipient => $masterTelnum,
  352.                    encode=>'chinese',
  353.                    status_report=>1,
  354.                    );
  355.                    if( $lOk ) {
  356.                           #
  357.                           #send ok to log
  358.                           #
  359.                           print "[group send]:\t--report is sent successful\n";
  360.                    } else {
  361.                           #
  362.                           #send error to log
  363.                           #
  364.                           print "[group send]:\t--report is sent fail\n";
  365.                    }
  366.     }
  367. }
复制代码

论坛徽章:
0
2 [报告]
发表于 2010-05-21 01:56 |只看该作者
本帖最后由 ykredrum 于 2010-05-21 02:00 编辑

一、部分代码解释:
my $masterTelnum = 134xxxxxxx7;
这句代码设定控制方手机号码,也就是你用这个号码发送命令接收发送和统计报告

our %team = (



      1588xxxxx17 => '张三',

      1343xxxxx37 => '李四,

);
这个hash存储的是群号码,key是手机号,value是姓名。这个hash的变量名很重要,发命令的时候要用。


my $gsm = new Device::Gsm( port => 'COM4', pin => $pin,log => 'file,send.log' );   
这个port的值要根据你蓝牙虚拟成串口的具体端口,直接改了就行。


二、运行方法:
    1.首先在电脑端运行程序。
    2.目前控制方有2个命令可以使用
      发送"status"   不包括"" 给短信机器人的号码,机器人将返回当前状态,在群发前最好先查看状态因为群发的时候可能对新收到的短信没有反应,单进程。
    3.发送 "team#2#大家好" 不包括""给短信机器人的号码,机器人将“大家好”三个字群发给team内的所有成员号码,发送完了将发送群发报告,而且如果2分钟内对方回短信的话,2分钟后将简单统计回复的内容并发送统计报告给控制方。这里的team就是上面所说的hash的变量名。


三、短信机器人的最初想法,
    主要想是用到球队,我们有个足球队,每次有比赛都要群发短信给队员有关球赛的相关信息,保证全部都要发送到,同时要统计哪些人能参加,哪些人不能参加,以便确定比赛时人能凑得够不,如果用手机发,我的手机群发一个组最多只能发20个人,而且中途要是有人发送失败,不能知道是哪个发送失败了,弄不好又要全部重发,再者就是发送完之后,紧接着就是一个接一个的短信回复,每个都要看下队员到底能参加不,现在这个程序做的事情就是,群发给n个,同时能生成发送报告,还有就是自动查看收到的短信,简单判断短信回复内容,最终统计回复内容,判断哪些能参加球赛,并将统计内容发给控制方手机。

头有点昏,不晓得表达清楚没有。代码有点垃圾,东拼西凑,只能说可以用。

论坛徽章:
1
未羊
日期:2014-09-08 22:47:27
3 [报告]
发表于 2010-05-21 08:21 |只看该作者
不错,高手。

论坛徽章:
0
4 [报告]
发表于 2010-05-21 08:35 |只看该作者
好,手机相关的东西,学习。

论坛徽章:
0
5 [报告]
发表于 2010-05-21 15:02 |只看该作者
忘了说了,原来的gsm.pm只能发英文的,后来在论坛里面找到有高手改成了支持中文的了!我直接拿来用了!
谢谢!

论坛徽章:
0
6 [报告]
发表于 2010-05-21 17:06 |只看该作者
路过。了解

论坛徽章:
0
7 [报告]
发表于 2010-05-22 09:34 |只看该作者
可惜了,是perl不是php的。

论坛徽章:
0
8 [报告]
发表于 2010-05-25 11:16 |只看该作者
强!!!也为诈骗犯提供了廉价的工具

论坛徽章:
5
技术图书徽章
日期:2014-02-10 10:55:18技术图书徽章
日期:2014-03-17 16:37:45狮子座
日期:2014-04-25 11:17:42未羊
日期:2014-08-13 11:45:23天蝎座
日期:2015-12-16 10:30:37
9 [报告]
发表于 2010-05-26 17:01 |只看该作者
gsm猫好贵的,要大几百,发送数量少的话不如以前的飞信脚本来得划算

论坛徽章:
0
10 [报告]
发表于 2010-05-29 22:33 |只看该作者
都支持哪些手机?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP