之前,我有写过一篇Blog关于相关问题的解决方案,文章地址。最近,在Windows下配置python-nmap时,又遇到了一些问题。总结记录一下。
关于”nmap program was not found in path”问题的解决办法:
首先,我本机安装了最新的Python-2.7.10和nmap-6.49BETA4。我还是打算用原来的python-nmap-0.2.4(官网最新版本已经到了python-nmap-0.4.0,关于它我们后续再说)。一切安装都OK之后,在命令行下运行一下python-nmap-0.2.4下的example.py文件,得到了如下的结果:
我去,怎么还是这个问题呢?参考前一篇文章的配置,结果毫无作用呀!什么鬼?!我修改了好多地方的配置,还是不知道什么情况。好吧,我最后决定在Pycharm调试运行一下,看看到底是什么鬼。调试关键行如下图所示:
从调试的结果看,应该是没有问题的,nmap_path已经赋值了。(突然,我就觉得似乎第一篇文章关于python-nmap问题中,提供的增加nmap的本地路径似乎也没啥必要嘛?!)接着往下看,看它到底在哪里抛出异常了。找到了,如下图所示:
这是什么?is_nmap_found默认为True,判断是否正则匹配,不匹配则赋值False。看来是正则匹配出问题了。原来,python-nmap判断是否安装了nmap的方法就是在找到的nmap执行路径下运行如下命令:
nmap -V
如果成功了,命令的返回值如下图所示:
注意!这里有个坑!现在,采用https协议的网站越来越多了。从返回结果看,提供的nmap官网地址所采用的也是https协议。问题来了,我们看下python-nmap-0.2.4中的正则匹配:
regex = re.compile('Nmap version [0-9]*\.[0-9]*[^ ]* \( http://nmap\.org \)')
竟然判断的是http协议有木有。这样,无论怎么都匹配不上,所以得到的判断条件总是False,最后也就抛出了异常,说找不到nmap。知道了这点就好办了,稍微修改一下nmap.py中的正则表达式,修改后的正则表达式如下所示:
regex = re.compile('Nmap version [0-9]*\.[0-9]*[^ ]* \( http(|s)://.* \)')
删除“C:\Python27\Lib\site-packages\nmap”下扩展名为pyc的文件,再次运行example.py,如下图所示:
一切正常了。如果使用的老版本的python-nmap,而nmap安装的又是最新版本的,就要注意”nmap -V”结果显示中使用的是https协议,而python-nmap中正则表达式判断的是http协议,稍微修改一下就OK了。
关于“close_fds is not supported on Windows platforms if you redirect stdin/stdout/stderr”问题的解决办法:
既然官网放出了最新的python-nmap-0.4.0版本,那就试用一下最新版本吧。安装一切OK之后,我来试一下,如下图所示:
我去,这就是个什么鬼?问题不断啊!冷静一下,我们来查下Python 2.7.10 documentation,看看subprocess.Popen这个函数是干啥的。感觉比较重要的部分在这里:
If close_fds is true, all file descriptors except 0, 1 and 2 will be closed before the child process is executed. (Unix only). Or, on Windows, if close_fds is true then no handles will be inherited by the child process. Note that on Windows, you cannot set close_fds to true and also redirect the standard handles by setting stdin, stdout or stderr.
难道是默认值的问题,我们来修改一下默认值,从True改为False试试:
p = subprocess.Popen([nmap_path, '-V'], bufsize=10000, stdout=subprocess.PIPE, close_fds=False)
再次运行,看一下:
不报错了,运行正常了。看来Linux下和Windows下还是有些不同的。所以在Windows下要使用python-nmap-0.4.0的话,记得把这个默认值改为False或者直接删掉函数申明中的赋值就好(因为默认值就是False)。
好了,就这么多,如果对你有用,那是最好不过的了。
|