11. Funções de rede do Python
Vamos agora discutir as funções de rede do Python, que nos permitem realizar programação TCP/IP (Protocolo de Controlo de Transmissão/Protocolo de Internet).
11.1. Obter o nome ou o endereço IP de um computador na Internet
Programa (inet_01)
| import sys, socket
#------------------------------------------------
def getIPandName(nomMachine):
# nomMachine: name of the machine whose address is required IP: name of the machine whose address is required IP: name of the machine whose address is required
# nomMachine-->adresse IP
try:
ip=socket.gethostbyname(nomMachine)
print "ip[%s]=%s" % (nomMachine,ip)
except socket.error, erreur:
print "ip[%s]=%s" % (nomMachine,erreur)
return
# address IP --> nomMachine
try:
name=socket.gethostbyaddr(ip)
print "name[%s]=%s" % (ip,name[0])
except socket.error, erreur:
print "name[%s]=%s" % (ip,erreur)
return
# ---------------------------------------- main
# constant
HOTES=["istia.univ-angers.fr","www.univ-angers.fr","www.ibm.com","localhost","","xx"]
# IP addresses of HOTES machines
for i in range(len(HOTES)):
getIPandName(HOTES[i])
# end
sys.exit()
|
Notas:
- Linha 1: As funções de rede do Python estão encapsuladas no módulo socket.
Resultados
| ip[istia.univ-angers.fr]=193.49.146.171
name[193.49.146.171]=istia.istia.univ-angers.fr
ip[www.univ-angers.fr]=193.49.144.40
name[193.49.144.40]=ametys-fo.univ-angers.fr
ip[www.ibm.com]=129.42.58.216
name[129.42.58.216]=[Errno 11004] host not found
ip[localhost]=127.0.0.1
name[127.0.0.1]=Gportpers3.ad.univ-angers.fr
ip[xx]=[Errno 11004] getaddrinfo failed
|
11.2. Um cliente web
Um script para recuperar o conteúdo da página de índice de um site.
Programa (inet_02)
| import sys,socket
#-----------------------------------------------------------------------
def getIndex(site):
# reads the URL site/ and stores it in the site.html file
# initially no error
erreur=""
# creation of the site.html file
try:
html=open("%s.html" % (site),"w")
except IOError, erreur:
pass
# mistake?
if erreur:
return "Erreur (%s) lors de la création du fichier %s.html" % (erreur,site)
# open a connection on site port 80
try:
connexion=socket.create_connection((site,80))
except socket.error, erreur:
pass
# return if error
if erreur:
return "Echec de la connexion au site (%s,80) : %s" % (site,erreur)
# connection represents a bidirectional communication flow
# between the client (this program) and the contacted web server
# this channel is used for the exchange of orders and information
# the dialog protocol is HTTP
# the customer sends the get command to request URL /
# syntax get URL HTTP/1.0
# protocol HTTP headers must end with an empty line
connexion.send("GET / HTTP/1.0\n\n")
# the server will now respond on the connection channel. It will send all
# then close the channel. The customer reads everything that comes in from the connection
# until the channel closes
ligne=connexion.recv(1000)
while(ligne):
html.write(ligne)
ligne=connexion.recv(1000)
# the customer in turn closes the connection
connexion.close()
# close html file
html.close()
# return
return "Transfert reussi du paragraphe index du site %s" % (site)
# --------------------------------------------- main
# get the text HTML from URL
# list of websites
SITES=("istia.univ-angers.fr","www.univ-angers.fr","www.ibm.com","xx")
# reading the index pages of the sites in the SITES table
for i in range(len(SITES)):
# read site index page SITES[i]
resultat=getIndex(SITES[i])
# result display
print resultat
# end
sys.exit()
|
Notas:
- linha 57: a lista de URLs dos sites cujas páginas de índice pretendemos. Esta está armazenada no ficheiro de texto [sitename.html];
- linha 62: a função getIndex realiza o trabalho;
- linha 4: a função getIndex;
- linha 20: o método create_connection((site,port)) permite criar uma ligação com um serviço TCP/IP em execução na porta port na máquina do site;
- linha 35: o método send permite que os dados sejam enviados através de uma ligação TCP/IP. Aqui, está a ser enviado texto. Este texto segue o protocolo HTTP (HyperText Transfer Protocol);
- linha 40: o método recv é utilizado para receber dados através de uma ligação TCP/IP. Aqui, a resposta do servidor web é lida em blocos de 1.000 caracteres e guardada no ficheiro de texto [sitename.html].
Resultados
Transfert reussi du paragraphe index du site istia.univ-angers.fr
Transfert reussi du paragraphe index du site www.univ-angers.fr
Transfert reussi du paragraphe index du site www.ibm.com
Echec de la connexion au site (xx,80) : [Errno 11001] getaddrinfo failed
O ficheiro recebido para o site [www.ibm.com]:
| HTTP/1.1 302 Found
Date: Wed, 08 Jun 2011 15:43:56 GMT
Server: IBM_HTTP_Server
Content-Type: text/html
Expires: Fri, 01 Jan 1990 00:00:00 GMT
Pragma: no-cache
Cache-Control: no-cache, must-revalidate
Location: http://www.ibm.com/us/en/
Content-Length: 209
Kp-eeAlive: timeout=10, max=14
Connection: Keep-Alive
<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML 2.0//EN">
<html><head>
<title>302 Found</title>
</head><body>
<h1>Found</h1>
<p>The document has moved <a href="http://www.ibm.com/us/en/">here</a>.</p>
</body></html>
|
- As linhas 1–11 são os cabeçalhos HTTP da resposta do servidor;
- linha 1: o servidor instrui o cliente a redirecionar para o URL especificado na linha 8;
- linha 2: data e hora da resposta;
- linha 3: identidade do servidor web;
- linha 4: conteúdo enviado pelo servidor. Aqui, uma página HTML que começa na linha 13;
- linha 12: a linha vazia que encerra os cabeçalhos HTTP;
- Linhas 13–19: a página HTML enviada pelo servidor web.
11.3. Um cliente SMTP
Entre os protocolos TCP/IP, o SMTP (Simple Mail Transfer Protocol) é o protocolo de comunicação para o serviço de entrega de mensagens.
Programa (inet_03)
| # -*- coding=utf-8 -*-
import sys,socket
#-----------------------------------------------------------------------
def getInfos(fichier):
# returns the information (smtp,sender,recipient,message) taken from the text file [file]
# line 1: smtp, sender, recipient
# next lines: message text
# open [file]
erreur=""
try:
infos=open(fichier,"r")
except IOError, erreur:
return ("Le fichier %s n'a pu etre ouvert en lecture : %s" % (fichier, erreur))
# read the 1st line
ligne=infos.readline()
# delete end-of-line mark
ligne=cutNewLineChar(ligne)
# retrieving smtp, sender and recipient fields
champs=ligne.split(",")
# do we have the right number of fields?
if len(champs)!=3 :
return ("La ligne 1 du fichier %s (serveur smtp, expediteur, destinataire) a un nombre de champs incorrect" % (fichier))
# "processing" the recovered information
# we remove from each of the 3 fields the "blanks" that precede or follow the useful information
for i in range(3):
champs[i]=champs[i].strip()
# field recovery
(smtpServer,expediteur,destinataire)=champs
message=""
# read rest of message
ligne=infos.readline()
while ligne!='':
message+=ligne
ligne=infos.readline()
infos.close()
# return
return ("",smtpServer,expediteur,destinataire,message)
#-----------------------------------------------------------------------
def sendmail(smtpServer,expediteur,destinataire,message,verbose):
# sends message to smtp server smtpserver from sender
# as recipient. If verbose=True, tracks client-server exchanges
# retrieve the customer's name
try:
client=socket.gethostbyaddr(socket.gethostbyname("localhost"))[0]
except socket.error, erreur:
return "Erreur IP / Nom du client : %s" % (erreur)
# open a connection on port 25 of smtpServer
try:
connexion=socket.create_connection((smtpServer,25))
except socket.error, erreur:
return "Echec de la connexion au site (%s,25) : %s" % (smtpServer,erreur)
# connection represents a bidirectional communication flow
# between the client (this program) and the smtp server contacted
# this channel is used for the exchange of orders and information
# after connection, the server sends a welcome message which is read as follows
erreur=sendCommand(connexion,"",verbose,1)
if(erreur) :
connexion.close()
return erreur
# cmde ehlo:
erreur=sendCommand(connexion,"EHLO %s" % (client),verbose,1)
if erreur :
connexion.close()
return erreur
# cmde mail from:
erreur=sendCommand(connexion,"MAIL FROM: <%s>" % (expediteur),verbose,1)
if erreur :
connexion.close()
return erreur
# cmde rcpt to:
erreur=sendCommand(connexion,"RCPT TO: <%s>" % (destinataire),verbose,1)
if erreur :
connexion.close()
return erreur
# cmde data
erreur=sendCommand(connexion,"DATA",verbose,1)
if erreur :
connexion.close()
return erreur
# prepare message to send
# it must contain the lines
# From: expéditeur
# To: recipient
# empty line
# Message
# .
data="From: %s\r\nTo: %s\r\n%s\r\n.\r\n" % (expediteur,destinataire,message)
# send message
erreur=sendCommand(connexion,data,verbose,0)
if erreur :
connexion.close()
return erreur
# cmde quit
erreur=sendCommand(connexion,"QUIT",verbose,1)
if erreur :
connexion.close()
return erreur
# end
connexion.close()
return "Message envoye"
# --------------------------------------------------------------------------
def sendCommand(connexion,commande,verbose,withRCLF):
# sends command to connection channel
# verbose mode if verbose=1
# if withRCLF=1, adds sequence RCLF to command
# data
RCLF="\r\n" if withRCLF else ""
# send cmde if order not empty
if commande:
connexion.send("%s%s" % (commande,RCLF))
# possible echo
if verbose:
affiche(commande,1)
# read response of less than 1000 characters
reponse=connexion.recv(1000)
# possible echo
if verbose:
affiche(reponse,2)
# error code recovery
codeErreur=reponse[0:3]
# error returned by the server?
if int(codeErreur) >=500:
return reponse[4:]
# error-free return
return ""
# --------------------------------------------------------------------------
def affiche(echange,sens):
# displays exchange ? screen
# if sens=1 displays -->change
# if sens=2 displays <-- exchange without last 2 characters RCLF
if sens==1:
print "--> [%s]" % (echange)
return
elif sens==2:
l=len(echange)
print "<-- [%s]" % echange[0:l-2]
return
# --------------------------------------------------------------------------
def cutNewLineChar(ligne):
# delete the [line] end-of-line mark if it exists
l=len(ligne)
while(ligne[l-1]=="\n" or ligne[l-1]=="\r"):
l-=1
return(ligne[0:l])
# main ----------------------------------------------------------------
# client SMTP (SendMail Transfer Protocol) for sending a message
# information is taken from a INFOS file containing the following lines
# line 1: smtp, sender, recipient
# next lines: message text
# sender:email sender
# recipient: email recipient
# smtp: name of smtp server to use
# communication protocol SMTP client-server
# -> client connects to smtp server port 25
# <- server sends him a welcome message
# -> customer sends command EHLO: machine name
# <- server responds OK or not
# -> customer sends mail from: <exp?diteur> command
# <- server responds OK or not
# -> client sends the rcpt to command: <recipient>
# <- server responds OK or not
# -> customer sends data order
# <- server responds OK or not
# -> client sends all the lines of its message and ends with a line containing the single character .
# <- server responds OK or not
# -> customer sends quit order
# <- server responds OK or not
# server responses have the form xxx text where xxx is a 3-digit number. Any number xxx >=500
# indicates an error. The answer may consist of several lines all beginning with xxx- except the last one
# of the form xxx(space)
# exchanged text lines must end with RC(#13) and LF(#10) characters
# # Mailing parameters
MAIL="mail2.txt"
# retrieve mail parameters
res=getInfos(MAIL)
# mistake?
if res[0]:
print "%s" % (erreur)
sys.exit()
# sending mail in verbose mode
(smtpServer,expediteur,destinataire,message)=res[1:]
print "Envoi du message [%s,%s,%s]" % (smtpServer,expediteur,destinataire)
resultat=sendmail(smtpServer,expediteur,destinataire,message,True)
print "Resultat de l'envoi : %s" % (resultat)
# end
sys.exit()
|
Notas:
- Num computador Windows com software antivírus, o antivírus pode impedir que o script Python se ligue à porta 25 de um servidor SMTP. Por isso, deve desativar o antivírus. No caso do McAfee, por exemplo, pode fazer o seguinte:
- Em [1], abra a consola do VirusScan
- Em [2], interrompa o serviço [On-Access Protection]
- em [3], o serviço está parado
Resultados
O ficheiro infos.txt:
smtp.univ-angers.fr, serge.tahe@univ-angers.fr , serge.tahe@univ-angers.fr
Subject: test
ligne1
ligne2
ligne3
Resultados do ecrã:
| Envoi du message [smtp.univ-angers.fr,serge.tahe@univ-angers.fr,serge.tahe@univ-angers.fr]
--> [EHLO Gportpers3.ad.univ-angers.fr]
<-- [220 smtp.univ-angers.fr ESMTP Postfix
250-smtp.univ-angers.fr
250-PIPELINING
250-SIZE 20480000
250-VRFY
250-ETRN
250-ENHANCEDSTATUSCODES
250-8BITMIME
250 DSN]
--> [MAIL FROM: <serge.tahe@univ-angers.fr>]
<-- [250 2.1.0 Ok]
--> [RCPT TO: <serge.tahe@univ-angers.fr>]
<-- [250 2.1.5 Ok]
--> [DATA]
<-- [354 End data with <CR><LF>.<CR><LF>]
--> [From: serge.tahe@univ-angers.fr
To: serge.tahe@univ-angers.fr
Subject: test
ligne1
ligne2
ligne3
.
]
<-- [250 2.0.0 Ok: queued as 63412114203]
--> [QUIT]
<-- [221 2.0.0 Bye]
Resultat de l'envoi : Message envoye
|
A mensagem tal como vista pelo cliente de e-mail Thunderbird:
11.4. Um segundo cliente SMTP
Este segundo script faz o mesmo que o anterior, mas utiliza as funcionalidades do módulo [smtplib].
Programa (inet_04)
| # -*- coding=utf-8 -*-
import sys,socket, smtplib
#-----------------------------------------------------------------------
def getInfos(fichier):
# returns the information (smtp,sender,recipient,message) taken from the text file [file]
# line 1: smtp, sender, recipient
# next lines: message text
# open [file]
erreur=""
try:
infos=open(fichier,"r")
except IOError, erreur:
return ("Le fichier %s n'a pu etre ouvert en lecture : %s" % (fichier, erreur))
# read the 1st line
ligne=infos.readline()
# delete end-of-line mark
ligne=cutNewLineChar(ligne)
# retrieve smtp, sender, recipient fields
champs=ligne.split(",")
# do we have the right number of fields?
if len(champs)!=3 :
return ("La ligne 1 du fichier %s (serveur smtp, expediteur, destinataire) a un nombre de champs incorrect" % (fichier))
# "processing" the information recovered - removing the "blanks" that precede or follow them
for i in range(3):
champs[i]=champs[i].strip()
# field recovery
(smtpServer,expediteur,destinataire)=champs
# read message to send
message=""
ligne=infos.readline()
while ligne!='':
message+=ligne
ligne=infos.readline()
infos.close()
# return
return ("",smtpServer,expediteur,destinataire,message)
#-----------------------------------------------------------------------
def sendmail(smtpServer,expediteur,destinataire,message,verbose):
# sends message to smtp server smtpserver from sender
# as recipient. If verbose=True, tracks client-server exchanges
# we use the smtplib library
try:
server = smtplib.SMTP(smtpServer)
if verbose:
server.set_debuglevel(1)
server.sendmail(expediteur, destinataire, message)
server.quit()
except Exception, erreur:
return "Erreur envoi du message : %s" % (erreur)
# end
return "Message envoye"
# --------------------------------------------------------------------------
def cutNewLineChar(ligne):
# delete the [line] end-of-line mark if it exists
l=len(ligne)
while(ligne[l-1]=="\n" or ligne[l-1]=="\r"):
l-=1
return(ligne[0:l])
# main ----------------------------------------------------------------
# client SMTP (SendMail Transfer Protocol) for sending a message
# information is taken from a INFOS file containing the following lines
# line 1: smtp, sender, recipient
# next lines: message text
# sender:email sender
# recipient: email recipient
# smtp: name of smtp server to use
# communication protocol SMTP client-server
# -> client connects to smtp server port 25
# <- server sends him a welcome message
# -> customer sends command EHLO: machine name
# <- server responds OK or not
# -> customer sends mail order from: <sender>
# <- server responds OK or not
# -> client sends the rcpt to command: <recipient>
# <- server responds OK or not
# -> customer sends data order
# <- server responds OK or not
# -> client sends all the lines of its message and ends with a line containing the single character .
# <- server responds OK or not
# -> customer sends quit order
# <- server responds OK or not
# server responses have the form xxx text where xxx is a 3-digit number. Any number xxx >=500
# indicates an error. The answer may consist of several lines all beginning with xxx- except the last one
# of the form xxx(space)
# exchanged text lines must end with RC(#13) and LF(#10) characters
# # Mailing parameters
MAIL="mail2.txt"
# retrieve mail parameters
res=getInfos(MAIL)
# mistake?
if res[0]:
print "%s" % (erreur)
sys.exit()
# sending mail in verbose mode
(smtpServer,expediteur,destinataire,message)=res[1:]
print "Envoi du message [%s,%s,%s]" % (smtpServer,expediteur,destinataire)
resultat=sendmail(smtpServer,expediteur,destinataire,message,True)
print "Resultat de l'envoi : %s" % (resultat)
# end
sys.exit()
|
Notas:
- Este script é idêntico ao anterior, exceto pela função sendmail. Esta função utiliza agora as funcionalidades do módulo [smtplib] (linha 3).
Resultados
O ficheiro mail2.txt:
smtp.univ-angers.fr, serge.tahe@univ-angers.fr , serge.tahe@univ-angers.fr
From: serge.tahe@univ-angers.fr
To: serge.tahe@univ-angers.fr
Subject: test
ligne1
ligne2
ligne3
Resultados do ecrã:
| Envoi du message [smtp.univ-angers.fr,serge.tahe@univ-angers.fr,serge.tahe@univ-
angers.fr]
send: 'ehlo Gportpers3.ad.univ-angers.fr\r\n'
reply: '250-smtp.univ-angers.fr\r\n'
reply: '250-PIPELINING\r\n'
reply: '250-SIZE 20480000\r\n'
reply: '250-VRFY\r\n'
reply: '250-ETRN\r\n'
reply: '250-ENHANCEDSTATUSCODES\r\n'
reply: '250-8BITMIME\r\n'
reply: '250 DSN\r\n'
reply: retcode (250); Msg: smtp.univ-angers.fr
PIPELINING
SIZE 20480000
VRFY
ETRN
ENHANCEDSTATUSCODES
8BITMIME
DSN
send: 'mail FROM:<serge.tahe@univ-angers.fr> size=99\r\n'
reply: '250 2.1.0 Ok\r\n'
reply: retcode (250); Msg: 2.1.0 Ok
send: 'rcpt TO:<serge.tahe@univ-angers.fr>\r\n'
reply: '250 2.1.5 Ok\r\n'
reply: retcode (250); Msg: 2.1.5 Ok
send: 'data\r\n'
reply: '354 End data with <CR><LF>.<CR><LF>\r\n'
reply: retcode (354); Msg: End data with <CR><LF>.<CR><LF>
data: (354, 'End data with <CR><LF>.<CR><LF>')
send: 'From: serge.tahe@univ-angers.fr\r\nTo: serge.tahe@univ-angers.fr\r\nSubje
ct: test\r\n\r\nligne1\r\nligne2\r\n\r\nligne3\r\n.\r\n'
reply: '250 2.0.0 Ok: queued as D4977114358\r\n'
reply: retcode (250); Msg: 2.0.0 Ok: queued as D4977114358
data: (250, '2.0.0 Ok: queued as D4977114358')
send: 'quit\r\n'
reply: '221 2.0.0 Bye\r\n'
reply: retcode (221); Msg: 2.0.0 Bye
Resultat de l'envoi : Message envoye
|
11.5. Echo cliente/servidor
Criamos um serviço de eco. O servidor devolve todas as linhas de texto enviadas pelo cliente em maiúsculas. O serviço pode atender vários clientes ao mesmo tempo utilizando threads.
O programa do servidor (inet_08)
| # -*- coding=utf-8 -*-
# generic tcp server on Windows
# loading header files
import re,sys,SocketServer,threading
# multi-threaded tcp server
class ThreadedTCPRequestHandler(SocketServer.StreamRequestHandler):
def handle(self):
# current thread
cur_thread = threading.currentThread()
# customer data
self.data="on"
# stop on empty chain
while self.data:
# the client's text lines are read using the readline method
self.data =self.rfile.readline().strip()
# console monitoring
print "client %s : %s (%s)" % (self.client_address[0],self.data,cur_thread.getName())
# reply to customer
response = "%s: %s" % (cur_thread.getName(), self.data.upper())
# self.wfile is the write stream to client
self.wfile.write(response)
class ThreadedTCPServer(SocketServer.ThreadingMixIn, SocketServer.TCPServer):
pass
# ------------------------------------------------------ main
# call syntax: argv[0] port
# the server is launched on the port named
# Data
syntaxe="syntaxe : %s port" % (sys.argv[0])
# ------------------------------------- check the call
# there must be one argument and one argument alone
nbArguments=len(sys.argv)
if nbArguments!=2:
print syntaxe
sys.exit(1)
# the port must be digital
port=sys.argv[1]
modele=r"^\s*\d+\s*$"
if not re.match(modele,port):
print "Le port %s n'est pas un nombre entier positif\n" % (port)
sys.exit(2)
# server launch - serves each client on a thread
host="localhost"
server = ThreadedTCPServer((host,int(port)), ThreadedTCPRequestHandler)
# the server is launched in a thread
# each customer will be served in an additional thread
server_thread = threading.Thread(target=server.serve_forever)
# starts the server - infinite waiting loop for clients
server_thread.start()
# follow-up
print "Serveur d'echo a l'ecoute sur le port % s" % (port)
|
Notas:
- linha 39: sys.argv representa os argumentos do script. Aqui, estes devem estar no formato: nome_do_script porta. Devem, portanto, ser dois. sys.argv[0] será então nome_do_script e sys.argv[1] será porta;
- linha 53: o servidor echo é uma instância da classe ThreadedTcpServer. O construtor desta classe espera dois parâmetros:
- parâmetro 1: uma tupla de dois elementos (host,port) que especifica a máquina e a porta de escuta do servidor;
- parâmetro 2: o nome da classe responsável por tratar os pedidos dos clientes.
- linha 57: é criada uma thread (mas ainda não iniciada). Esta thread executa o método [serve_forever] do servidor TCP. Este método é um loop que escuta as ligações dos clientes. Assim que for detetada uma ligação de um cliente, será criada uma instância da classe ThreadedTCPRequestHandler. O seu método handle é responsável pela comunicação com o cliente;
- Linha 59: A thread do serviço de eco é iniciada. A partir deste ponto, os clientes podem ligar-se ao serviço;
- linha 27: a classe do servidor de eco. Ela deriva de duas classes:
*SocketServer.ThreadingMixIn* e SocketServer.TCPServer. Isto torna-a um servidor TCP multithread: o servidor é executado numa thread e cada cliente é atendido numa thread adicional;
- Linha 9: a classe que lida com os pedidos dos clientes. Ela deriva da classe SocketServer.StreamRequestHandler. Assim, herda dois atributos:
- rfile: que é o fluxo de leitura para os dados enviados pelo cliente — pode ser tratado como um ficheiro de texto;
- wfile: que é o fluxo de escrita utilizado para enviar dados ao cliente — pode ser tratado como um ficheiro de texto.
- linha 11: o método handle processa as solicitações do cliente;
- linha 13: a thread que executa este método handle;
- linha 17: loop para processar as solicitações do cliente. O loop termina quando o cliente envia uma linha vazia;
- linha 19: lê a solicitação do cliente;
- linha 21: self.client_address[0] representa o endereço IP do cliente. cur_thread.getName() é o nome da thread que executa o método handle;
- linha 23: a resposta ao cliente tem dois componentes — o nome da thread que atende o cliente e o comando enviado pelo cliente, convertido para maiúsculas.
O programa cliente (inet_06)
| # -*- coding=utf-8 -*-
import re,sys,socket
# ------------------------------------------------------ main
# generic tcp client
# call syntax: argv[0] host port
# client connects to echo service (host,port)
# the server returns the lines typed by the client
# syntax
syntaxe="syntaxe : %s hote port" % (sys.argv[0])
# ------------------------------------- check the call
# there must be two arguments
nbArguments=len(sys.argv)
if nbArguments!=3:
print syntaxe
sys.exit(1)
# we recover the arguments
hote=sys.argv[1]
# the port must be digital
port=sys.argv[2]
modele=r"^\s*\d+\s*$"
if not re.match(modele,port):
print "Le port %s foit être un nombre entier positif" % (port)
sys.exit(2)
try:
# client connection to server
connexion=socket.create_connection((hote,int(port)))
except socket.error, erreur:
print "Echec de la connexion au site (%s,%s) : %s" % (hote,port,erreur)
sys.exit(3)
try:
# input loop
ligne=raw_input("Commande (rien pour arreter): ").strip()
while ligne!="":
# send the line to the server
connexion.send("%s\n" % (ligne))
# we're waiting for the answer
reponse=connexion.recv(1000)
print "<-- %s" % (reponse)
ligne=raw_input("Commande (rien pour arreter): ")
except socket.error, erreur:
print "Echec de la connexion au site (%s,%s) : %s" % (hote,port,erreur)
sys.exit(3)
finally:
# close the connection
connexion.close()
|
Resultados
O servidor é iniciado numa janela de comando:
cmd>%python% inet_08.py 100
Serveur d'echo a l'ecoute sur le port 100
Um primeiro cliente é iniciado numa segunda janela:
| cmd>%python% inet_06.py localhost 100
Commande (rien pour arreter): cmde 1 du client 1
[cmde 1 du client 1]
<-- Thread-2: CMDE 1 DU CLIENT 1
Commande (rien pour arreter): cmde 2 du client 1
<-- Thread-2: CMDE 2 DU CLIENT 1
Commande (rien pour arreter):
|
O cliente recebe, em resposta ao comando que envia ao servidor, esse mesmo comando em maiúsculas. Um segundo cliente é iniciado numa terceira janela:
| cmd>%python% inet_06.py localhost 100
Commande (rien pour arreter): cmde 1 du client 2
<-- Thread-3: CMDE 1 DU CLIENT 2
Commande (rien pour arreter): cmde 2 du client 2
<-- Thread-3: CMDE 2 DU CLIENT 2
Commande (rien pour arreter):
|
A consola do servidor fica então assim:
| cmd>%python% inet_08.py 100
Serveur d'echo a l'ecoute sur le port 100
client 127.0.0.1 : cmde 1 du client 1 (Thread-2)
client 127.0.0.1 : cmde 2 du client 1 (Thread-2)
client 127.0.0.1 : cmde 1 du client 2 (Thread-3)
client 127.0.0.1 : cmde 2 du client 2 (Thread-3)
|
Os clientes são atendidos corretamente em threads diferentes. Para interromper um cliente, basta introduzir um comando vazio.
11.6. Servidor TCP genérico
Propomos escrever um script em Python que
- funcione como um servidor TCP capaz de atender um cliente de cada vez,
- aceite linhas de texto enviadas pelo cliente,
- aceite linhas de texto do teclado e as reenvie ao cliente.
Assim, o utilizador ao teclado atua como servidor:
- vê as linhas de texto enviadas pelo cliente no seu terminal;
- responde ao cliente digitando a resposta no teclado.
Pode, assim, adaptar-se a qualquer tipo de cliente. É por isso que o chamaremos de «servidor TCP genérico». É uma ferramenta prática para explorar protocolos de comunicação TCP. No exemplo seguinte, o cliente TCP será um navegador web, o que nos permitirá explorar o protocolo HTTP utilizado por clientes web.
O programa (inet_10)
| # -*- coding=utf-8 -*-
# generic tcp server
# loading header files
import re,sys,SocketServer,threading
# generic tcp server
class MyTCPHandler(SocketServer.StreamRequestHandler):
def handle(self):
# the customer is displayed
print "client %s" % (self.client_address[0])
# create a thread for reading customer orders
thread_lecture = threading.Thread(target=self.lecture)
thread_lecture.start()
# stop on cmde 'bye
# cmde reading typed on keyboard
cmde=raw_input("--> ")
while cmde!="bye":
# send cmde to customer. self.wfile is the write flow to the customer
self.wfile.write("%s\n" % (cmde))
# next cmde reading
cmde=raw_input("--> ")
def lecture(self):
# displays all lines sent by the client until the bye command is received
ligne=""
while ligne!="bye":
# the client's text lines are read using the readline method
ligne = self.rfile.readline().strip()
# console monitoring
print "<--- %s : %s" % (self.client_address[0], ligne)
# ------------------------------------------------------ main
# call syntax: argv[0] port
# the server is launched on the port named
# Data
syntaxe="syntaxe : %s port" % (sys.argv[0])
# ------------------------------------- check the call
# there must be one argument and one argument alone
nbArguments=len(sys.argv)
if nbArguments!=2:
print syntaxe
sys.exit(1)
# the port must be digital
port=sys.argv[1]
modele=r"^\s*\d+\s*$"
if not re.match(modele,port):
print "Le port %s n'est pas un nombre entier positif\n" % (port)
sys.exit(2)
# server launch
host="localhost"
server = SocketServer.TCPServer((host, int(port)), MyTCPHandler)
print "Service tcp generique lance sur le port %s. Arret par Ctrl-C" % (port)
server.serve_forever()
|
Notas:
- Linha 57: O servidor TCP será uma instância da classe SocketServer.TCPServer. O seu construtor aceita dois parâmetros:
- o primeiro parâmetro é uma tupla de dois elementos (host, port), em que host é a máquina na qual o serviço é executado (normalmente localhost) e port é a porta na qual o serviço aguarda (escuta) os pedidos dos clientes;
- o segundo parâmetro especifica a classe de serviço do cliente. Quando um cliente se liga, é criada uma instância da classe de serviço, e o seu método handle deve gerir a ligação com o cliente.
O Tcp SocketServer.TCPServer não é multithread. Atende um cliente de cada vez;
- linha 59: o método
serve\_forever do servidor Tcp é executado. Trata-se de um ciclo infinito que aguarda clientes;
- linha 9: o servidor TCP aqui é uma classe derivada da classe SocketServer.StreamRequestHandler. Isto permite que os fluxos de dados trocados com o cliente sejam tratados como ficheiros de texto. Já nos deparámos com esta classe. Temos os seguintes métodos:
- *
readline* para ler uma linha de texto do cliente;
write para enviar linhas de texto de volta ao cliente.
- Linha 12:
*self.client\_address[0]* é o endereço IP do cliente;
- linha 14: o servidor TCP irá comunicar com o cliente utilizando duas threads
- um thread para ler linhas do cliente;
- uma thread para escrever linhas para o cliente.
- linha 14: é criada a thread para ler linhas do cliente. O seu parâmetro
target especifica o método executado pela thread. Este é o definido na linha 25;
- Linha 25: A thread de leitura é iniciada;
- Linhas 25–32: O método executado pela thread de leitura;
- linha 28: o thread de leitura lê todas as linhas de texto enviadas pelo cliente até que a linha «bye» seja recebida;
- linha 18: estamos agora na thread que escreve para o cliente. A ideia é enviar ao cliente todas as linhas de texto digitadas pelo utilizador no teclado.
Os resultados
O servidor
dos>%python% inet_10.py 100
Service tcp generique lance sur le port 100. Arret par Ctrl-C
O navegador do cliente
O pedido recebido pelo servidor
| dos>%python% inet_10.py 100
Service tcp generique lance sur le port 100. Arret par Ctrl-C
client 127.0.0.1
<--- 127.0.0.1 : GET / HTTP/1.1
<--> --- 127.0.0.1 : Host: localhost:100
<--- 127.0.0.1 : User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; fr; rv:1.9.2.10) Gecko/20100914 Firefox/3.6.10 GTB7.1 ( .NET CLR 3.5.30729)
<--- 127.0.0.1 : Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
<--- 127.0.0.1 : Accept-Language: fr,fr-fr;q=0.8,en;q=0.5,en-us;q=0.3
<--- 127.0.0.1 : Accept-Encoding: gzip,deflate
<--- 127.0.0.1 : Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.7
<--- 127.0.0.1 : Keep-Alive: 115
<--- 127.0.0.1 : Connection: keep-alive
<--- 127.0.0.1 : Cache-Control: max-age=0
<--- 127.0.0.1 :
|
Resposta do servidor digitada pelo utilizador no teclado (sem o sinal -->)
| HTTP/1.1 200 OK
--> Server: serveur tcp generique
--> Connection: close
--> Content-Type: text/html
-->
--> <html><body><h2>Serveur tcp generique</h2></body></html>
--> bye
|
- Linhas 1-5: Resposta HTTP enviada ao cliente;
- linha 6: Página HTML enviada ao cliente;
- Linha 7: Fim do diálogo com o cliente. O serviço do cliente será encerrado e a ligação será fechada, o que interromperá abruptamente o segmento de código que lê as linhas de texto enviadas pelo cliente;
- As linhas 1–5 da resposta HTTP enviada ao cliente têm os seguintes significados:
- Linha 1: O recurso solicitado pelo cliente foi encontrado;
- Linha 2: Identificação do servidor;
- Linha 3: O servidor encerrará a ligação após enviar o recurso;
- Linha 4: Tipo de recurso enviado pelo servidor: um documento HTML;
- Linha 5: uma linha vazia.
A página apresentada pelo navegador [1]:
Se analisarmos o código-fonte recebido pelo navegador [2], podemos ver que o código HTML recebido pelo navegador é, de facto, aquele que lhe enviámos.