泰国游记

2008年菲律宾2009年台湾省2010年美国硅谷2011年美国硅谷,最近几年每年都有一次出境的机会,运气还不错。前几天公司组织了一次以自费为主的泰国游,行程总得来说不是很有意思,就在这里分享一下这次行程的一些Tips和体会,供大家参考。

本次行程主要包含泰国曼谷和芭提雅两个城市,总共在泰国呆5个白天,由于是跟团游,总的感觉是所有的时间都被旅行社占用掉了,完全没有自主的时间。这点对于喜欢乱逛的我来说是很郁闷的,尤其是没有机会去体验曼谷市区的轨道交通,颇为遗憾。

在出发前,建议在国内提前到银行预约泰铢的兑换,在国内兑换的汇率比在泰国要合算的多,最不济的情况下,在泰国的ATM机上用银联卡直接提款也比在机场或导游那里兑换要合算。泰国银联的普及率还是比较高的,大部分情况下用银联卡比走VISA/MasterCard是要合算的。

泰国天气较热,阳光也比较强烈,最好准备SPF指数较高的防晒产品。跟其它国家一样,酒店大多不提供一次性用品,自己带上比较靠谱。泰国电压跟中国一样AC220V 50Hz,插座大多是美标的三脚插座,两脚插头的设备都没有压力,三脚的可能需要转换器。

泰国洒店大多不提供免费Wi-Fi和Internet服务,可以付费使用但价格较高,或者可以在泰国购买手机卡打电话或上网。中国移动和中国电信的泰国漫游资费都比较高,GSM和CDMA网络泰国都有,不用换手机。

泰国有遍地的7-Eleven便利店,在那里可以买到日常用品和手机卡。便利店的店员不保证能听懂中文或英文,但酒店和旅游场所的人大多会一点中文或英文。

TrueMove运营商为游客设计的手机卡49泰铢一张,另外可以用充值卡充值50或90铢。打中国的电话是1.07铢/分钟,上网可以用50铢包50M的流量。以我个人的经历,TrueMove的网络很不怎么样,常常上不了网,而且扣费也扣得莫明其妙。客服电话是收费的,而且只有泰文提示音,我是乱按了好久才转接到人工服务的,人工服务可以用英语。所以如果要用TrueMove,应该在国内时先上网查好它的各项服务的使用方法。

泰国服务业普遍有收小费的习惯,一般情况下可以给20泰铢左右。

我的这次行程景点主要包括曼谷的玉佛寺、大皇宫、湄南河、大城王朝遗迹(维修中)、芭堤雅水上集市、金沙岛和五世皇行宫,总体来说没有感觉看到有特别让人Wow的东西。进入皇家的建筑要求穿长裤和有袖上衣,不然就只能花钱买一条布把自己围起来了。

导游带的泰国购物点主要包括珠宝中心、皮件中心、燕窝中心、干果市场、毒蛇研究中心和免税店。某种程度上说宝石、鳄鱼皮、燕窝、水果干和蛇药确实可以算是泰国的特产,是值得买的,但对于导游带去的购物点,尤其是珠宝中心,还是尽量要谨慎一些,如果确实要买,建议提前做功课。我对于干果市场的水果干品质表示极度怀疑,所以后来特地偷闲跑到曼谷当地一家很大的Tesco Lotus超市去,结果发现这么大的综合超市中完全不见水果干这种商品,7-Eleven/Big C便利店也没有,看来水果干这个东西是“特供”外国游客了。在泰国购买欧莱雅的化妆品相比国内比较有优势,其它的都明显比美国或香港要贵得多。Lee的牛仔裤大约2000铢左右,跟美国或菲律宾比也很没有优势。

最具泰国特色的人妖表演或相关的“自费”项目,实话说没有什么特别值得一看,大多所谓表演很拙劣,不具有艺术性,可以映证那句“不看后悔,看了也后悔”的话。相对来说在芭提雅艺城看的泰国历史文化演出还比较有意思一点,不过这个如果不提前做点功课了解一点泰国历史的话, 看不懂。

泰国很多景点都有一个不错的营销模式,在游客进入景区时,会有人以迅雷不及掩耳之势为各位游客拍摄照片。等游客游完景点后,这些照片已经冲印完毕并嵌入相框,或做成盆子、杯子之类的工艺品,放在景点门口以100-200铢的价格出售。尽管大家都自己带着相机拍照,但看到这些成品的照片还是会有很多人愿意花钱买下来。这种“所见即所得”的营销模式,感觉比在国内常见的搭个大阳伞被动等人花钱来拍照的的效果要好得多。

泰国新鲜的热带水果还是很便宜的,椰子即使在景区宰客的情况下,也只有20-25铢一个。山竹、榴莲、红毛丹和芒果都很值得推荐 。对了,还有DQ的冰淇凌也很便宜,小杯的只要29铢。

大部分照片都按惯例放在Picasa相册(包括Google+)中了,有兴趣的自行设法去看吧:

体验WPS for Linux

昨天得知WPS for Linux开始发放测试邀请,马上去申请了一下,今天拿到了邀请码,回到家速度展开测试。

这次放出的WPS for Linux实际上只包含了Kingsoft Presentation这一个组件,以DEB包的方式发布,122M大,不算小。

安装:

我所使用的系统是Debian, i686架构,使用GNOME桌面, WPS for Linux的安装包在我的环境中缺少ttf-mscorefonts-installer这个依赖,这个不难解决,apt-get一下就行。之所以要依赖这些Windows下字体,我想可能是为了在不同的平台下提供一个比较一致的体验吧。

解决依赖问题后,用dpkg -i直接安装WPS for Linux,报了一堆错:

/usr/bin/xdg-mime: 769: cannot create /root/.local/share/applications/defaults.list.new: Directory nonexistent
grep: /root/.local/share/applications/defaults.list.new: No such file or directory

不知道为什么安装程序非要不依不饶往/root/.local中去注册MIME类型,写/usr/local/share中不好么?反正都给了root权限了。不管怎么说,帮它创建好一个/root/.local/share/applications目录,然后重新安装,安装成功。

运行:

安装后,在菜单中Office项中,会出现Kingsoft Presentation项,点了点,没有反应。出现这种现象常常是因为库的问题,所以去命令行运行wpp:

/opt/kingsoft/wps-office/office6/wpp: /usr/lib/libstdc++.so.6: version `GLIBCXX_3.4.14' not found (required by /opt/kingsoft/wps-office/office6/libkso10.so.1)

好吧,libstdc++库的问题,我的Debian Squeeze中带的版本太低了。在Debian网站上找一个sid中的版本,直接装是不敢啦,怕把系统搞挂了(实际上也装不上,因为别的依赖的问题),所以用dpkg-deb -x把它解开,然后用加上LD_LIBRARY_PATH的方式运行,WPS for Linux的Presentation界面终于出现在我的眼前了。

WPS for Linux Presentation主界面

WPS for Linux Presentation主界面

体验:

WPS Presentation提供了两种风格的界面,包括类似于Office 2003的经典界面和类似于Office 2007+的Ribbon界面。可以通过“切换界面”按钮或菜单项完成切换。

WPS Presentation支持的文件格式不多,只有它自己的.dps格式和Office的.ppt/.pptx格式。.odp格式没有被支持这点不是很美好,也很出乎我的意料。它也没有提供导出.pdf的功能,也不是很美好。

作为WPS一贯的作风,它对Office文件的兼容性还是相当不错的,打开.ppt文件展现出来效果跟Windows下比并没有严重的走样。

保存文件的功能在我的环境中有严重的问题,会直接导致程序core掉,不知道跟我简单偷换libstdc++.so有没有关系。跟locale有关,用dpkg-reconfigure locales加上zh_CN.UTF-8的locale后,问题就消失了。原本我机器上只有en_US.UTF-8的locale。

WPS Presentation在首页上提供了从金山网站在线下载幻灯片模版的功能,很好用,有些模版也比较好看。相对而言,Office提供的这样的功能我就从来都没用过。

从使用感觉上来说,跟被我淘汰了有一阵子了的OpenOffice.org和正在使用的LibreOffice相比,WPS给我留下的最直接的印象是:快。整个软件在我的机器上表现非常流畅,用起来很舒服,这点很让我满意。

WPS没有提供英文拼写检查的功能,不过作为一个面向国人的软件来说,不算是太大的问题。

软件的稳定性似乎还存在一定的问题,在试用过程中,发生过若干次莫名其妙的crash,但暂时还没有找到稳定重现的方法。

体验完了WPS Presentation,期待WPS的Word Processing System早日登场。

PS. 突然有点怀念下面的这个界面:

UCDOS下的WPS启动界面

UCDOS下的WPS启动界面

还有多少人记得Ctrl + q i u b o j u n是用来做什么的?

 

偷梁换柱终级版:用最少的步骤在Bambook上阅读原生EPUB文件

本文内容已过时,云梯0.98b版本开始,直接支持向Bambook中传送原生EPUB格式文件,不再需要用本文的方法去折腾了。

我不得不说,伸手党是促进社会劳动生产率不断进步的源泉。

前一篇“偷梁换柱”写完以后放到某论坛上,很快就有伸手党跳出来说,还是太复杂了,能不能把所有的东西写成一个软件一运行就好了。我有点生气的回复他,可以,不过我没有兴趣和精力做这个。不过过了一会儿转念想想,其实这事儿吧,也真没有那么难。花了几十分钟,搞定了这个“终级版”的“偷梁换柱”,希望后面我不要再想出什么更好的办法来,不然我真不知道该用什么词来做文章的标题了。

目标:把calibre中管理的自有EPUB格式的电子书传到Bambook中原生阅读,不做格式转换。

适用人群:Bambook用户,有用Bambook阅读原生EPUB格式电子书的需求,基本会用calibre管理电子书,会用记事本修改文本文件,能知道自己所用电脑的IP地址。

系统与环境要求:安装过云梯和calibre的Windows操作系统,Bambook通过USB连接,可以通过该Windows系统代理上网。或者安装过calibre的Windows操作系统,Bambook通过Wi-Fi上网。(其它操作系统的用户,也可以参照执行,完全是可行的,只是本文中不做针对性说明,如果有问题,可以留言询问)

1. 从本站下载工具包:

本地下载:工具包

把下载的文件解压缩到硬盘上,比如C:\,得到fakedns.py和fakehttp.py。还有两个批处理1.bat和2.bat。

2. 运行脚本文件

如果calibre是安装在默认的C:\Program Files\Calibre2中,直接双击运行1.bat和2.bat即可。运行成功的话,会在屏幕上保留两个命令提示符的窗口,里面会显示本机IP,后面还会不断显示一些调试信息。

如果calibre不是装在默认路径下:用记事本打开1.bat和2.bat,修改里面的calibre安装路径后再运行。

两个脚本运行后,都会在屏幕上打出本机的IP地址。如果与实际情况不符,请把命令提示符窗口关掉,用记事本修改脚本文件中的IP = “”行,写成IP = “192.168.0.1”这样的形式,强行指定本机IP,保存后重新运行这两个脚本。

3. 修改电脑或Bambook的DNS设置,指向我们的假DNS服务器

为了确保成功,在这里先把Bambook断开USB或无线连接,并重启一次。

如果用电脑USB共享给Bambook上网,就修改电脑上网网卡上的DNS设置,改成本机的IP地址(在第2步的脚本运行过程中会打印在屏幕上,照抄就行了)。改完后,把Bambook用USB连接到电脑上。

如果Bambook自己Wi-Fi无线上网,就修改Bambook上网络配置中的DNS设置,设成电脑的IP地址。改完后,关闭并重新开启Bambook的无线连接。

在修改前,请记下原来的值(也有可能原来选的是“自动获取”),以便在用完后把它恢复原样。

4. 启动calibre服务器

运行calibre,主界面下“连接/共享”->“启动连接服务”,这时calibre应该会在8080端口上启动http服务器,提供 OPDS服务,可以通过访问http://127.0.0.1:8080/opds来验证。(这是在默认配置的情况下,如果有改变过配置,需修改fakehttp.py中的URL)

搞定了!

在Bambook上按“找书”->1.访问云中书城->*.切换到第三方书库,或者最新固件上书架主菜单中选3.更多书城,就可以看到被我们 “篡改”过的第三方书城列表,选择其中的5. calibre,就可以尽情的享用你本地calibre书库中的EPUB图书了。

从calibre中获取到心仪的EPUB图书后,记得把第3步中改过的DNS值恢复原样,免得影响正常上网。

偷梁换柱升级版:用calibre为Bambook提供EPUB书库

本文内容已过时,云梯0.98b版本开始,直接支持向Bambook中传送原生EPUB格式文件,不再需要用本文的方法去折腾了。

上一篇Blog中提供了在Bambook上阅读EPUB书籍一个折腾方法,但那个方法显然很tricky,依赖环境多,成功率低,要传多本图书更复杂。这次又折腾出个改进版的方法,虽然也比较复杂,但成功率比以前高多了,而且也更直观好用,分享给大家。原理:用calibre提供的OPDS服务,把这个服务器添加到Bambook的第三方书城列表中。用到的软件和技术跟以前差不多:calibre,DNS劫持,nginx反向代理与rewrite。

前提条件:有一台正常上网的电脑A,Bambook用无线上网并在上网过程中可以访问到电脑A。或者,有两台电脑(可以是虚拟机)A和B,A用来做服务器,B上安装云梯给Bambook共享网络,AB之间也可以相互访问。

1. 在电脑A上下载安装nginx for Windows

http://nginx.org/download/nginx-1.1.15.zip

下载后,解压缩到C:\,得到C:\nginx-1.1.15

2. 配置nginx,实现Web服务和反向代理

用记事本打开C:\nginx-1.1.15\conf\nginx.conf,把它修改成以下内容,其中的192.168.8.102为A电脑的IP地址,请自行替换(下同)

worker_processes  1;
events {
    worker_connections  1024;
}

http {
    include       mime.types;
    default_type  application/octet-stream;
    server {
        listen          80;
        server_name     opds.gk.sdo.com;
        location / {
            proxy_pass              http://$http_host$request_uri;
            proxy_redirect          off;
            proxy_set_header Host   $host;
            proxy_set_header        X-Real-IP       $remote_addr;
            proxy_set_header        X-Forwarded-For $proxy_add_x_forwarded_for;
            rewrite ^.*Default.aspx.*$ http://192.168.8.102:81/1.xml redirect;    
        }
    }
    server {
        listen       81;
        location / {
            root   html;
            index  index.html index.htm;
        }
    }           
    sendfile        on;
    keepalive_timeout  65;
}

3. 在C:\nginx-1.1.15\html中创建1.xml文件,内容如下,使用UTF-8编码(如果用记事本,那就需要用“另存为”对话框,把下面的“编码”由ANSI改为UTF-8)。

<?xml version="1.0" encoding="utf-8"?>
<feed xmlns:opds="http://opds-spec.org/2010/catalog" xmlns:opensearch="http://a9.com/-/spec/opensearch/1.1/" xmlns:dcterms="http://purl.org/dc/terms/" xmlns:thr="http://purl.org/syndication/thread/1.0" xmlns:dc="http://dublincore.org/documents/dcmi-namespace/" xmlns="http://www.w3.org/2005/Atom">
  <title type="text">锦书OPDS源列表</title>
  <id>uuid:ea8a979d-7e28-4bf7-8d11-31aafb2ac66a;id=60</id>
  <updated>2012-03-11T15:49:25Z</updated>
  <link rel="search" type="application/atom+xml" href="http://opds.gk.sdo.com/Search.aspx?ST=ALL&amp;PageIndex=1&amp;k={searchTerms}&amp;{bbtk}" />
  <entry>
    <id>JiuYue</id>
    <title type="text">新华书店九月网</title>
    <updated>2012-03-11T15:49:25Z</updated>
    <link rel="alternate" type="application/atom+xml" href="http://opds.9yue.com/category.atom?site=BamBook" />
    <content type="text">新华书店九月网</content>
  </entry>
  <entry>
    <id>http://m.gutenberg.org/ebooks/?format=opds</id>
    <title type="text">古腾堡计划</title>
    <updated>2012-03-11T15:49:25Z</updated>
    <link rel="alternate" type="application/atom+xml" href="http://m.gutenberg.org/ebooks/?format=opds" />
    <content type="text">英文电子书</content>
  </entry>
  <entry>
    <id>http://www.feedbooks.com/site/free_books.atom</id>
    <title type="text">feedbooks</title>
    <updated>2012-03-11T15:49:25Z</updated>
    <link rel="alternate" type="application/atom+xml" href="http://www.feedbooks.com/site/free_books.atom" />
    <content type="text">在线阅读网线图书</content>
  </entry>
  <entry>
    <id>http://www.feedbooks.com/site/free_books.atom</id>
    <title type="text">calibre</title>
    <updated>2012-03-11T15:49:25Z</updated>
    <link rel="alternate" type="application/atom+xml" href="http://192.168.8.102:8080/opds" />
    <content type="text">calibre</content>
  </entry>
</feed>

4. 启动nginx

双击运行C:\nginx-1.1.15\nginx.exe,用任务管理查看nginx进程,确认启动成功。如果用netstat -na检查,应该在80和81端口上有LISTENING。如果访问http://192.168.8.102:81/1.xml,可以看到上面我们建立的1.xml文件。

5. 安装Python 2.7

http://www.python.org/getit下载Python Windows Installer并安装到C:\Python27目录。

6. 写Python脚本,建假DNS。把下面的内容存为C:\FakeDNS.py。同样,其中的192.168.8.102为A电脑的IP地址,请自行替换

#! /usr/bin/env python
# This code comes from
# http://code.activestate.com/recipes/491264-mini-fake-dns-server/
# with some modifications
import socket

class DNSQuery:
  def __init__(self, data):
    self.data=data
    self.domain=''
    tipo = (ord(data[2]) >> 3) & 15
    if tipo == 0:
      ini=12
      lon=ord(data[ini])
      while lon != 0:
        self.domain+=data[ini+1:ini+lon+1]+'.'
        ini+=lon+1
        lon=ord(data[ini])

  def respuesta(self, ip):
    packet=''
    if self.domain:
      packet+=self.data[:2] + "\x81\x80"
      packet+=self.data[4:6] + self.data[4:6] + '\x00\x00\x00\x00'
      packet+=self.data[12:]
      packet+='\xc0\x0c'
      packet+='\x00\x01\x00\x01\x00\x00\x00\x3c\x00\x04'
      packet+=str.join('',map(lambda x: chr(int(x)), ip.split('.')))
    return packet

if __name__ == '__main__':
  udps = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
  udps.bind(('',53))
  try:
    while 1:
      data, addr = udps.recvfrom(1024)
      p = DNSQuery(data)
      if p.domain == 'opds.gk.sdo.com.':
        ip = '192.168.8.102'
      else:
        ip = socket.gethostbyname(p.domain)
      udps.sendto(p.respuesta(ip), addr)
      print p.domain + "=>" + ip
  except KeyboardInterrupt:
    udps.close()

7. 启动假DNS

在命行提示符中运行:

C:\Python27\python.exe C:\FakeDNS.py

启动后应该会停在那里,有请求时会打印出解析结果。可以再开一个窗口用nslookup确认它的工作状态:对于除了opds.gk.sdo.com以外的域名,都解析为正常IP地址;对于opds.gk.sdo.com.,解析为电脑A的IP地址。

8. Bambook设置

设置Bambook的无线连接,其中DNS服务器地址设为假DNS所在电脑A的IP地址。

如果采用两台电脑的方案,就把电脑B的网卡配置中的DNS指向电脑A的IP地址。然后连接Bambook与电脑B,让Bambook通过电脑B上网。

9. 启动calibre服务器

运行calibre,主界面下“连接/共享”->“启动连接服务”,这时calibre应该会在8080端口上启动http服务器,提供OPDS服务,可以通过访问http://127.0.0.1:8080/opds来验证。(这是在默认配置的情况下,如果有改变过配置,上面的1.xml做相应改动即可)

一切就绪,在Bambook上按“找书”->1.访问云中书城->*.切换到第三方书库,或者最新固件上书架主菜单中选3.更多书城,就可以看到被我们“篡改”过的第三方书城列表,选择其中的5. calibre,就可以尽情的享用你本地calibre书库中的EPUB图书了。

注意:这个方法只支持Bambook从calibre书库中下载EPUB格式的图书。如果书城列表获取不正常,可能是DNS缓存的问题,重启Bambook后再试一次。

calibre除了自带的OPDS连接服务,也可以用calibre2opds这个软件来提供更美好的OPDS服务器,以后另文介绍calibre2opds。

偷梁换柱:在Bambook中阅读自有epub电子书

本文内容已过时,云梯0.98b版本开始,直接支持向Bambook中传送原生EPUB格式文件,不再需要用本文的方法去折腾了。

先写一句话指南,免得浪费有很强动手能力的技术青年们的时间:通过假DNS把res.gk.sdo.com劫持到本地,对于非.epub文件下载请求,反向代理到真正的res.gk.sdo.com上;对于.epub文件请求,你想把什么书传到Bambook上,就在本地把这个.epub通过HTTP返回出去吧。在Linux上,可以用nginx做Web服务器、反向代理和Rewrite,用dnsmasq或我下面提供的Python脚本做假DNS。

我也算是Bambook的老用户了,从它上市第一天起就开始使用,当年写的体验心得算是很给面子、非常委婉指出了它的一些致命缺点(对于我自己而言的),其中最重要的两条是:只有Windows-Only的客户端程序和不支持除了它自有的SNB以外的任何电子书格式。第一个问题,我通过盛大后来开放的SDK给calibre写了一些插件基本解决了(支持Windows/Linux/Mac OS X,使用方法在这里),第二个问题盛大后来自己开始支持PDF格式了,算是解决了一半,但如果再能支持一下epub格式就美好了。

在盛大去年发布的最后一版支持第三方书库(Feedbooks和古登堡计划)的固件中,某些人敏锐的注意到,从第三方书库下载的电子书明显有比SNB格式更丰富的排版,而且下载页面上也以很小的字体提示了说,这是epub格式的。由此可知,Bambook的固件已经可以支持epub的直读,剩下的问题就是怎么把epub书放进去了。

尝试过一些方法,比如把epub文件跟pdf文件一样,用SNB格式包裹起来再通过SDK传到Bambook上,总之都是失败。最后实在没有办法,决定用Wireshark抓包分析Bambook从Feedbooks下载图书的交互过程,从中做点坏事。

分析过程其实一点也不复杂,会用Wireshark的人应该都能搞定。我选了对Feedbooks的交互过程进行分析,分析的结果是,Bambook从第三方书库下载图书时,仍然会与盛大的服务器交互,并且最终是从盛大的服务器的URL上下载实际的文件(最新的分析结果是:可能从盛大的服务器,也可能从原始feedbooks的服务器,取决于盛大的服务器有没有缓存过这个文件。该分析未完全确定),所以简单的劫持对Feedbooks.com的访问是没用的。反复尝试,下面描述的方法是可行的(但很可能不是最简单的,懒得再深入研究了)。为照顾大部分Windows用户,下面以Windows环境来介绍。Linux用户请参考本文第一段自己折腾,应该更简单一些。

前提条件:有一台正常上网的电脑A,Bambook用无线上网并在上网过程中可以访问到电脑A。或者,有两台电脑(可以是虚拟机)A和B,A用来做服务器,B上安装云梯给Bambook共享网络,AB之间也可以相互访问。

1. 在电脑A上下载安装nginx for Windows

http://nginx.org/download/nginx-1.1.15.zip

下载后,解压缩到C:\,得到C:\nginx-1.1.15

2. 配置nginx,实现Web服务和反向代理

用记事本打开C:\nginx-1.1.15\conf\nginx.conf,把它修改成以下内容,其中的192.168.8.102为A电脑的IP地址,请自行替换

worker_processes  1;
events {
    worker_connections  1024;
}

http {
    include       mime.types;
    default_type  application/octet-stream;
    server {
        listen          80;
        server_name     res.gk.sdo.com;
        location / {
            proxy_pass        http://res.gk.cdn.sdo.com/;
            proxy_redirect    off;
            proxy_set_header  Host            $host;
            proxy_set_header  X-Real-IP       $remote_addr;
            proxy_set_header  X-Forwarded-For $proxy_add_x_forwarded_for;
            rewrite ^.*/.*\.epub$ http://192.168.8.102:81/1.epub redirect;
        }
    }
    server {
        listen       81;
        location / {
            root   html;
            index  index.html index.htm;
        }
    }           
    sendfile        on;
    keepalive_timeout  65;
}

3. 启动nginx

双击运行C:\nginx-1.1.15\nginx.exe,用任务管理查看nginx进程,确认启动成功。如果用netstat -na检查,应该在80和81端口上有LISTENING。

4. 安装Python 2.7

http://www.python.org/getit下载Python Windows Installer并安装到C:\Python27目录。

5. 写Python脚本,建假DNS。把下面的内容存为C:\FakeDNS.py。同样,其中的192.168.8.102为A电脑的IP地址,请自行替换

#! /usr/bin/env python
# This code comes from
# http://code.activestate.com/recipes/491264-mini-fake-dns-server/
# with some modifications
import socket

class DNSQuery:
  def __init__(self, data):
    self.data=data
    self.domain=''
    tipo = (ord(data[2]) >> 3) & 15
    if tipo == 0:
      ini=12
      lon=ord(data[ini])
      while lon != 0:
        self.domain+=data[ini+1:ini+lon+1]+'.'
        ini+=lon+1
        lon=ord(data[ini])

  def respuesta(self, ip):
    packet=''
    if self.domain:
      packet+=self.data[:2] + "\x81\x80"
      packet+=self.data[4:6] + self.data[4:6] + '\x00\x00\x00\x00'
      packet+=self.data[12:]
      packet+='\xc0\x0c'
      packet+='\x00\x01\x00\x01\x00\x00\x00\x3c\x00\x04'
      packet+=str.join('',map(lambda x: chr(int(x)), ip.split('.')))
    return packet

if __name__ == '__main__':
  udps = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
  udps.bind(('',53))
  try:
    while 1:
      data, addr = udps.recvfrom(1024)
      p = DNSQuery(data)
      if p.domain == 'res.gk.sdo.com.':
        ip = '192.168.8.102'
      else:
        ip = socket.gethostbyname(p.domain)
      udps.sendto(p.respuesta(ip), addr)
      print p.domain + "=>" + ip
  except KeyboardInterrupt:
    udps.close()

6. 启动假DNS

在命行提示符中运行:

C:\Python27\python.exe C:\FakeDNS.py

启动后应该会停在那里,有请求时会打印出解析结果。可以再开一个窗口用nslookup确认它的工作状态:对于除了res.gk.sdo.com以外的域名,都解析为正常IP地址;对于res.gk.sdo.com,解析为电脑A的IP地址。

7. Bambook设置

设置Bambook的无线连接,其中DNS服务器地址设为假DNS所在电脑A的IP地址。

如果采用两台电脑的方案,就把电脑B的网卡配置中的DNS指向电脑A的IP地址。然后连接Bambook与电脑B,让Bambook通过电脑B上网。

8. 把打算要传到Bambook上的EPUB电子书放到C:\nginx-1.1.15\html中,命名为1.epub。

9. Bambook上选择Feedbooks.com上任意一本Public Domain的图书,下载阅读。

找书->1.访问云中书城->*.切换到第三方书库->3.feedbooks->3.Public Domain Books->2.Recently Added->任选一本以前没有下载过的书->1. 免费试读本书。

搞定!返回书架后就可以看到你自己的epub电子书已经放进了Bambook,可以尽情的阅读了。

重要更新:如果有时候发现下载下来的还是第三方书库的书,不是被自己替换过的。也许是因为盛大服务器上还没缓存过这本书,所以我们劫持盛大服务器就没用了。暂时的解决方案是换一本排在比较前面的书(这些书有缓存的可能性比较大),或者到书架上把刚刚下载的书删掉,重新再去下同一本,那时这书可能已经被缓存在盛大的服务器上了,这样我们的劫持就有效了。

其它说明:

1. 不能只用一台电脑或不用Bambook无线上网搞定的原因是:Bambook USB共享上网中的NAT服务中带了DNS服务,会与我们的假DNS抢53端口。

2. 制作在Bambook上阅读的EPUB电子书时,内部单个文件不要太大,不然翻页速度会让你抓狂。如果用calibre转换格式,可以在EPUB格式输出插件的设置中设定页面文件分割大小。

3. 虽然我们偷换掉了EPUB文件,但似乎Bambook还是认为我们是下载了我们所选的书籍并会记住这个状态,再次选择同一本书时,就不会再下载了。所以每次传书都得从Feedbooks中找一本以前没有用过书才行。

4. 这个方法只在Bambook SD928经典白色版上验证有效,未在全键盘版上验证过。且盛大只要服务器那里稍微动动手脚就可以让它失效,所以不保证有效期。并欢迎大家寻求更简单完美的解决方案。

5. 我曾经有过另一个在Bambook上阅读富格式书籍的想法,就是把HTML(或EPUB、MOBI等,反正本质都是HTML)页面封装成Bambook的Widget,用JavaScript在HTML中实现阅读器应该有的功能。现在是有SDK可以把Widget传入Bambook的,所以用这个方法完全可以写软件(或者给calibre写个插件,非常简单)自动完成常见的电子书格式转成Widget和传入Bambook的全过程。这个方法我已经做过POC,是完全可行的。但我没有能力用Java Script写出一个让我自己满意的阅读程序,所以最终放弃了。

6. 本文描述的所有内容仅供技术宅折腾和探讨,不愿折腾的人、Kindle或其它功能强大的阅读器拥护者请勿喷。谢谢。