校内网狗狗刷骨机的ruby实现

      前些天上网看到有人的博客上写了怎么来给校内网狗狗刷骨头,是用perl写的,现在我用ruby简单实现以下。原perl实现代码在http://www.cnblogs.com/xiaoyz/archive/2008/10/27/1320259.html

 

当然,由于校内网也发现了相关问题,会经常换狗狗flash,md5_key,所以这个必须先反编译swf文件找到对应的代码才会知道。

 

注意: 先刷狗粮(就是earnFoods),刷到很多了之后用狗粮给被人狗喂食(这个时候将planA.earnFoods前面加上'#',并把planA.earnBones前面的'#'去掉),那就会赠送骨头给你,这样就是刷骨头了,骨头多了后随便买东西,然后继续刷狗粮,刷骨头,刷狗粮,刷骨头。。。

 

另:校内网很多东西再不断变化,如果服务器端代码变化导致用不了,那就别用了,或者用我下面写的方法将相关改变的东西也改过去。

 

 运行代码前,先将你对应的fid,pid,tsc,cookie 等四个信息填入下面代码。

 

#Author:http://fortianwei.iteye.com
#Date:2009-4-18
#
#
require 'uri'
require 'net/http'
require 'md5'
require 'rexml/document'
class XiaoNeiDog
  attr_reader :key,:conf
  @@flashAddress='http://dog.xnimg.cn/dog/xn_swf/dogs_v0.14.swf'#这个随时会更改
  @@MD5_key='%$T&*jkhjksdhfjk$%^&*BJHhh'  #这个不好弄,要反编译出来,随时会更改
  @@tsc='XXXXXXXXXXXXX'#这个用网络嗅探器得到数据包中获取
  @@fid='XXXXXXXXXXX'#用户id
  @@pid='XXXXXXXXXXx'#狗狗id
  @@storeAddress='http://dog.xiaonei.com/store.do'
  @@cookie='XXXXXXXXXXXXXXXXXXXX' #cookie
  @@head={"Cookie"=>"#{@@cookie}",
          "Accept-Language"=>"zh-CN",
          "x-flash-version"=>"9,0,124,0",
          "Referer"=>"#{@@flashAddress}"
    }

  #初始化
  #获得初始化的狗狗key,做任何事都要获得狗狗的key,这个key是随机数
  #所以每次都要记录,方便下次动作来用,初始化的xml配置单和以后临时配置单数据格式不一样
  #所以初始化不用给@conf赋值
def initialize
     init
  end
  
  def init
      url="/pet-profile.do?method=dogXML&actor_id=" +@@fid + "&pid=" + @@pid
      link=Net::HTTP.new('dog.xiaonei.com',80)
      result=link.get(url,@@head)
      xml=result.body
      doc=REXML::Document.new xml
      @key=REXML::XPath.first(doc,'//key').text
      #bones=REXML::XPath.first(doc,'//num_bones_available').text
      #foods=REXML::XPath.first(doc,'//food_available').text
      #puts "food_available:"<<foods
      #puts "bones_available:"<<bones
  end

#返回狗狗临时配置单的REXML::Document对象
def getXMLConfObject
  REXML::Document.new @conf
end

 #通过狗狗配置单获得狗狗每次随机key
def getKeyFromXML
   getXMLConfObject.root.attributes['key']
end

#看狗狗渴了没
def thirsty?
     getXMLConfObject.root.attributes['is_pet_thirsty']=="true"
end

#看狗狗饿了没 不是看配置单中的is_pet_hungry而是判断狗狗体力是否达到85%,可以把函数改名叫forceOut?
def forceOut?
   getXMLConfObject.root.attributes['force'].to_f/getXMLConfObject.root.attributes['max_force'].to_f<0.85 ? true: false
end

#获取挨饿的狗狗们的id
def getHungryDogs
  s=Set.new
  url='http://dog.xiaonei.com/hall.do?method=hungry'
  link=Net::HTTP.new("dog.xiaonei.com",80)
  result=link.get(url,@@head).body.to_s
  dogsId=result.scan(/pid=\d+/i).each do    
  |dogId| dogId=dogId.sub!(/pid=/,'')
end
sleep(5)
s.union dogsId
end

 #用每次不同的狗狗配置单输出狗狗骨头和狗粮信息
 def outputDogInfo
   foods,bones=getXMLConfObject.root.attributes['user_foods'],getXMLConfObject.root.attributes['user_bones']
   puts "pid :"<<@@pid 
   puts "food_available:"<<foods
   puts "bones_available:"<<bones
  
 end
 
  #陪狗狗玩
  def playWithDog
    begin
    puts "===============play with dog=============="
       code = @key + "#" + @@pid + "#" + @@fid+"#"+@@MD5_key
       
       secret="actor%5Fid=" + @@fid + "&method=pelt&code="+ MD5.hexdigest(code) + "&pid=" + @@pid
       url="/pet-profile.do"
       link=Net::HTTP.new("dog.xiaonei.com",80)
       result=link.post(url,secret,@@head)
       @conf=result.body
       @key=getKeyFromXML#每次玩完后要设定@key的值
       outputDogInfo
    rescue
       puts @conf
       #earnFoods
    ensure
       return @conf
    end
  end
     
  #狗狗体力不支,吃点东西喝点水吧,"获得体力"
  #type=1的时候是吃狗粮,type=2的时候是喝水
  def feachForce(type)
      code=@key+"#"+@@pid+"#"+@@fid+"#"+@@MD5_key
      secret="active%5Fid="+type.to_s+ "&method=doing&actor%5Fid="+@@fid+"&code="+ MD5.hexdigest(code) + "&pid=" + @@pid
      url="/pet-profile.do"
      link=Net::HTTP.new("dog.xiaonei.com",80)
      result=link.post(url,secret,@@head)
      @conf=result.body
      puts @conf
      @key=getKeyFromXML#每次玩完后要设定@key的值
      foods=type==1? "foods":"water"
      puts "=========feed dog #{@@pid} with #{foods}================="
      outputDogInfo
      
  end
  
  #骨头够5个了,去买狗粮吧,免得等会输回去了^_^,,不过第一步是获取一个叫做tsc的用作store key的东东
  #这个要先进去商店,从返回的html里面获取有一个tsc=xxxxxxx的东东就是,不过用网络嗅探器更好得到
  def buyFoods
      puts "===============buy foods==============="
      url="/store.do"
      link=Net::HTTP.new("dog.xiaonei.com",80)
      result=link.post(url,"method=buyfood&buy_id=1&buy_count=1&tsc="+@@tsc,{"Cookie"=>"#{@@cookie}","Referer"=>"http://dog.xiaonei.com/store.do?method=food"})
  end

  #赚狗粮,也就是玩球球,看概率,超过5个骨头马上换狗粮
  def earnFoods
    begin
    count=0
     while(true) do
     count+=1
        playWithDog
        puts @conf
        sleep(6)
        buyFoods if getXMLConfObject.root.attributes['user_bones'].to_i>=5
        feachForce(2) if thirsty?
        feachForce(1) if forceOut?
        
       # return if count==1200
        end
      rescue 
        puts "error"
     #   earnFoods
    end
  end
  
  #赚骨头,方法:给别人挨饿的狗狗喂食,好人有好报啊
  #不过一般吃狗粮前要喝水
  def earnBones
    while true do
      list=getHungryDogs
      begin
      list.each do
        |dogId| @@pid=dogId 
        init
        sleep(5)
        puts "feed dog #{@@pid}"
        feachForce(2)
        sleep(5)
        feachForce(1)
      end
      rescue Timeout::Error
       puts "Time out this time!"
      earnBones 
    end
  end
  
  end
  
end

planA=Internet.new
planA.earnFoods
#planA.earnBones

 顺带说一下那些不好得到的东东怎么搞到的:

先贴一个网络嗅探器的截图:

 

1.

 @@flashAddress='http://dog.xnimg.cn/dog/xn_swf/dogs_v0.14.swf'#这个随时会更改

你用的那个狗狗的flash,就是你再给狗丢球玩之前发送的数据包上图中的Referer后面对应网址,当然,上面只是一个数据包示范而已,真实的由于包含本人信息就没放上来。

 

2.

@@tsc='f8490cc8bfad54d7ca148cf81b82cec0'#这个用网络嗅探器得到数据包中获取
  @@fid='XXXXXXXXXXX'#用户id
  @@pid='XXXXXXXXXXx'#狗狗id
 @@cookie='XXXXXXXXXXXXXXXXXXXX' #cookie

这些比如fid,pid都可以从cookie中找到,而@@cookie就是那个cookie的全部,复制过来就行。而tsc是在你进入商店后(方式是post)生成的数据包中传递的数据中得到比如我的就是method=buyfood&buy_id=1&buy_count=1&tsc=f8490cc8bfad54d7ca148cf81b82cec0

 

3.

@@MD5_key='%$T&*jkhjksdhfjk$%^&*BJHhh'  #这个不好弄,要反编译出来,随时会更改

这个的确不好弄,先要用swf反编译器得到源代码,然后我是先查看到有一个Pokey文件里面有下面代码:

var _loc_6:* = current_state;
            _loc_6.ping_count = current_state.ping_count++;
            var _loc_2:* = new URLVariables();
            _loc_2.actor_id = INFO["actor_id"];
            _loc_2.active_id = String(param1);
            _loc_2.pid = INFO["pet_id"];
            var _loc_3:* = code + "#" + INFO["pet_id"] + "#" + INFO["actor_id"] + "#" + 

ConfigurationManager.MD5_KEY;
            _loc_2.code = MD5.hash(_loc_3);
            _loc_2.method = "doing";






            var _loc_4:* = new URLLoader();
            new URLLoader().addEventListener(Event.COMPLETE, handlePingResponse);
            var _loc_5:* = new URLRequest(ConfigurationManager.PING_URL);
            new URLRequest(ConfigurationManager.PING_URL).data = _loc_2;
            _loc_5.method = URLRequestMethod.POST;
            _loc_4.load(_loc_5);

 上面代码中很明显的加了一个ConfigurationManager.MD5_KEY,这个是关键(别的比如method="doing"表示是吃东西或喝水,怎么得到的?从网络嗅探器中得知)。那么这个ConfigurationManager.MD5_KEY怎么得到?我看了看有个叫做com.xiaonei.pokey.managers::ConfigurationManager的包,到里面一看,有如下代码。

package com.xiaonei.pokey.managers
{

    public class ConfigurationManager extends Object
    {
        static public const PING_URL:String;
        static public const FLASH_SHOP:String;
        static public const MD5_KEY:String = "%$T&*jkhjksdhfjk$%^&*BJHhh";
        static public const HOUSE_URL:String;
        static public const APP_URL:String = "http://dog.xiaonei.com/";


 很显然,这只是个头部,还有别的就不用贴上来了,我们可以看到

static public const MD5_KEY:String = "%$T&*jkhjksdhfjk$%^&*BJHhh";

那就是我代码里面对应的@@MD5_key='%$T&*jkhjksdhfjk$%^&*BJHhh'

不过,这个很容易改动,所以当你用我这个代码用不了的时候,可能就是这个值又变化了。

 

 

 

 

你可能感兴趣的:(.net,xml,Flash,perl,Ruby)