2011-06-12 54 views
3

前一段时间,我写了一封小小的WOL脚本[来自Google的帮助]打开我网络中的计算机。这里是脚本:来自外部WAN的WOL

exec /usr/bin/python -x "$0" "[email protected]" 
# 
node_lst = [ 
     'srv1 0a:1b:8c:0d:2e:7f', 
     'srv2 0A-0B-4C-8D-CE:3F', 
] 
# 
import os,sys,string,commands 
import struct, socket 
import re,random 

retval = 0 

mac_addr = "mac_addr.txt" 
X = '([a-zA-Z0-9]{2}[:|\-|.]?){5}[a-zA-Z0-9]{2}' 
S = re.compile(r'\s+') 

mmap = {} 

## First argument 'None' in str.translate is new in 2.6. 
## Previously, it was a string of 256 characters 
if sys.version_info < (2, 6): 
    f1_arg = ''.join(chr(i) for i in xrange(256)) 
else: 
    f1_arg = None 

## broadcast address 
sysOS = "uname -s" 
BSD = "ifconfig | grep -w broadcast | cut -d\ -f 6" 
LNX = "ip -o addr show | grep -w inet | grep -e eth | cut -d\ -f 9" 
# 
if commands.getoutput(sysOS) == "Linux": 
    bCast = commands.getoutput(LNX) 
elif commands.getoutput(sysOS) == "Darwin": 
    bCast = commands.getoutput(BSD) 
else: 
    print "System not supported!!" 
    sys_exit() 

def WakeOnLan(mac_address): 

    ## Building the Wake-On-LAN "Magic Packet"... 
    ## Pad the synchronization stream. 
    data = ''.join(['FFFFFFFFFFFF', mac_address * 20]) 
    msg = '' 

    ## Split up the hex values and pack. 
    for i in range(0, len(data), 2): 
     msg = ''.join([msg, struct.pack('B', int(data[i: i + 2], 16))]) 

    ## ...and send it to the broadcast address using UDP 
    s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) 
    s.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1) 
    s.sendto(msg, (bCast, 9)) 
    s.close() 

def sys_exit(): 
    sys.stdout.flush() 
    sys.exit(1) 

## check if hostname is provided 
if len(sys.argv) != 2: 
    print "Usage: %s <hostname>" % sys.argv[0] 
    sys_exit() 

for i in node_lst: 
    # strip off everything from first "#" [if] found 
    i = i.split('#',1)[0] 
    if not re.search(X, i): 
     continue 

    h = S.split(i,1)[0]     ## host name 
    m = S.split(i,1)[-1]    ## MAC address 
    mmap[h] = m.strip('\t|" "') 

for j, k in mmap.iteritems(): 
    if sys.argv[1] == j: 
     if not re.search(X.replace('zA-Z','fA-F'), k): 
      print "Invalid MAC address [",k,"]; nothing to do!!" 
      sys_exit() 
     else: 
      WakeOnLan(k.translate(f1_arg,':.-')) 
      print "WOL request has been sent to %s [%s]" % (j,k) 
      break 
else: 
    print "Host [%s] doesn't exist!!" % sys.argv[1] 
    sys_exit() 

这在我的家庭网络(或局域网)内工作得很好。如何更改脚本以使其在局域网之外工作?任何想法或建议?干杯!!

回答

1

这是不可能的,因为WOL数据包是广播数据包(因为你不知道是谁发送它)。家庭路由器,尤其是ISP /网络路由器会丢弃所有的广播数据包,因为每次运行这个脚本时,整个互联网上的所有计算机都会收到你的数据包,这会造成一些混乱。

你当然可以做的是编写一个小型应用程序,该应用程序位于广域网内运行的计算机上,您希望打开所有计算机,然后让该应用程序发送WOL数据包。但是,这需要一台可以随时上网的电脑。

+0

我在互联网上发现了几篇文章,说可能使用路由器的IP地址,并使用NAT /端口转发到一个不存在的局域网IP地址。干杯!! – MacUsers 2011-06-12 20:08:43

+0

如果你确实知道如何,请让每个人都知道:)。 – 2011-06-12 20:09:22

+0

这里有一个,刚刚发现(我应该在发布之前进行搜索和尝试):'http:// www.schuetzler.net/blog/4/first-python-program-wake-on-wan'还没有测试过。 – MacUsers 2011-06-12 20:50:45

1

配置您的路由器将10个非顺序端口选择的数据包转发到LAN上的一台计算机。

根据格林尼治标准时间+时间+哈希来设计一些方案来生成端口触发序列。

在您的网络内的命令框中有一个python程序(使用scappy)监听一系列syn数据包。

监听器代码将类似于tcpdump的以下语法:

sudo tcpdump -ni eth0 'tcp[tcpflags] & (tcp-syn) !=0' 

凡刚捕获的SYN数据包。

你的程序就在那里,等待正确的顺序。当它收到序列时,它会运行你的WOL脚本。

完成。

如果您不想打开端口,您的脚本可以改为轮询远程网站,等待更改。或者收听通过电子邮件获取的电子邮件。

将你的想法进一步发挥,你可以做一些花哨的东西,比如开灯或启动电视。

+0

听起来很有希望。我会试试这个报告。干杯!! – MacUsers 2013-04-15 11:09:09