Python的socket模块为操作系统的socket实现提供了一个python接口。
In [168]: import socket
In [169]: s=socket.socket()
In [170]: s.connect(("127.0.0.1",80))
In [171]: s.send("GET / HTTP/1.1\n\n")
Out[171]: 16
In [172]: s.recv(200)
Out[172]: 'HTTP/1.1 400 Bad Request\r\nServer: nginx\r\nDate: Mon, 21 Jul 2014 02:34:14 GMT\r\nContent-Type: text/html\r\nContent-Length: 166\r\nConnection: close\r\n\r\n<html>\r\n<head><title>400 Bad Request</title></head>\r\n<b'
In [173]: s.close()
先创建一个名为s的socket对象,然后连接到本地的Nginx服务器,这里要注意一下connect只能带一个参数,如果是IP和端口对需要用括号括住。然后通过send()函数发送一个HTTP请求,然后接收服务端响应的前200个字节。最后关闭socket连接
#/usr/bin/python
import socket
import re
import sys
def check_server(address, port):
s=socket.socket()
print "Attempting to connect to %s on port %s" %(address,port)
try:
s.connect((address,port))
print "Connected to %s on port %s" %(address,port)
return True
except socket.error,e:
print "Connection to %s on port %s failed: %s" %(address,port,e)
return False
if __name__ == '__main__':
from optparse import OptionParser
parser = OptionParser()
parser.add_option("-a", "--address", dest="address",default='localhost', help="Address for the server", metavar="ADDRESS")
parser.add_option("-p", "--port", dest="port",type="int", default=80,help="PORT for server",metavar="PORT")
(options, args) = parser.parse_args()
print 'options: %s,args: %s' %(options,args)
check=check_server(options.address,options.port)
print 'check server returned %s' %check
sys.exit(not check)
以上程序用于探测远程主机的端口状态,通过socket的connect()函数去尝试连接远程主机的给定端口。后面主要用到optparse模块添加选项。
$ python tcp_port_checker.py -a 10.10.41.20 -p 80
options: {'port': 80, 'address': '10.10.41.20'},args: []
Attempting to connect to 10.10.41.20 on port 80
Connected to 10.10.41.20 on port 80
check server returned True
$ python tcp_port_checker.py -a 10.10.41.20 -p 80 && echo "SUCCESS"
options: {'port': 80, 'address': '10.10.41.20'},args: []
Attempting to connect to 10.10.41.20 on port 80
Connected to 10.10.41.20 on port 80
check server returned True
SUCCESS
$ python tcp_port_checker.py -a 10.10.41.20 -p 81 || echo "FAILURE"
options: {'port': 81, 'address': '10.10.41.20'},args: []
Attempting to connect to 10.10.41.20 on port 81
Connection to 10.10.41.20 on port 81 failed: [Errno 111] Connection refused
check server returned False
FAILURE
一个web server的80端口开放并不能代表这台服务器上就运行着web服务,还需要通过HTTP响应返回值来判断。
#/usr/bin/python
import socket
import re
import sys
def check_webserver(address,port,resource):
if not resource.startswith('/'):
resource = '/' + resource
request_string = "GET %s HTTP/1.1\r\nHost: %s\r\n\r\n" %(resource,address)
print 'HTTP request:'
print '|||%s|||' %request_string
s=socket.socket()
print "Attempting to connect to %s on port %s" %(address,port)
try:
s.connect((address,port))
print "Connected to %s on port %s" %(address,port)
s.send(request_string)
rsp=s.recv(100)
print "Received 100 bytes of HTTP response"
print '|||%s|||' %rsp
except socket.error,e:
print "Connection to %s on port %s failed: %s" %(address,port,e)
print "Closing the connection"
s.close()
lines=rsp.splitlines()
print "First line of HTTP response: %s" %lines[0]
try:
version,status,message=re.split(r'\s+',lines[0],2)
print 'Version: %s, Status: %s, Message: %s' %(version,status,message)
except ValueError:
print 'Failed to split status line'
return False
if status in ['200','301']:
print 'Success - status was %s' %status
return True
else:
print 'Status was %s' %status
return False
if __name__ == '__main__':
from optparse import OptionParser
parser = OptionParser()
parser.add_option("-a","--address",dest="address",default='localhost',help="ADDRESS for webserver",metavar="ADDRESS")
parser.add_option("-p","--port",dest="port",type="int",default=80,help="PORT for webserver",metavar="PORT")
parser.add_option("-r","--resource",dest="resource",default='index.html',help="RESOURCE for webserver",metavar="RESOURCE")
(options,args)=parser.parse_args()
print 'options: %s,args: %s' %(options,args)
check=check_webserver(options.address,options.port,options.resource)
print 'check_webserver returned %s' %check
sys.exit(not check)
以上程序主要通过check_webserver函数去探测HTTP服务,检测端口是否可以连接,并发送HTTP请求,通过web server的返回值进行判断。
$ python tcp_port_checker2.py -a 10.10.41.20 -p 80 -r index.php
options: {'resource': 'index.php', 'port': 80, 'address': '10.10.41.20'},args: []
HTTP request:
|||GET /index.php HTTP/1.1
Host: 10.10.41.20
|||
Attempting to connect to 10.10.41.20 on port 80
Connected to 10.10.41.20 on port 80
Received 100 bytes of HTTP response
|||HTTP/1.1 404 Not Found
Server: nginx
Date: Tue, 22 Jul 2014 03:24:17 GMT
|||tent-Type: text/html
Closing the connection
First line of HTTP response: HTTP/1.1 404 Not Found
Version: HTTP/1.1, Status: 404, Message: Not Found
Status was 404
check_webserver returned False