前些天上网看到有人的博客上写了怎么来给校内网狗狗刷骨头,是用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'
不过,这个很容易改动,所以当你用我这个代码用不了的时候,可能就是这个值又变化了。