#!/usr/bin/python # -*- coding: utf-8 -*- # exemple pyDuino - par X. HINAULT - www.mon-club-elec.fr # Juillet 2013 - Tous droits réservés - GPLv3 # voir : https://github.com/sensor56/pyDuino # Tester l'envoi d'une requete Ajax, l'envoi d'une reponse et la gestion d'une reponse Ajax from pyduino import * # importe les fonctions Arduino pour Python import numpy as np # entete declarative compt=0 # variable de comptage ipLocale=Ethernet.localIP() # auto - utilise l'ip de l'interface eth0 du systeme #ipLocale='192.168.1.25' # manuel - attention : utiliser la meme IP qu'une interface reseau du systeme # pour connaitre les interfaces reseau sur le systeme : utiliser la commande $ ifconfig print ipLocale # affiche l'adresse IP port=8080 # attention port doit etre au dessus de 1024 sinon permission refusee par securite - 8080 pour http serverHTTP=EthernetServer(ipLocale, port) # crée un objet serveur utilisant le port 8080 = port HTTP > 1024 data=None # Tableau Numpy global des donnees #--- setup --- def setup(): # -- serveur TCP -- global serverHTTP, ipLocale, port #serverHTTP.begin(10) # initialise le serveur - fixe nombre max connexion voulu serverHTTP.begin() # initialise le serveur - nombre max connexion par defaut = 5 print ("Serveur TCP actif avec ip : " + ipLocale + " sur port : " + str(port) ) #--- fin setup # -- loop -- def loop(): global serverHTTP print ("Attente nouvelle connexion entrante...") clientDistant, ipDistante = serverHTTP.clientAvailable() # attend client entrant # code bloque ici tant que pas client ! Si present, on recupere d'un coup objet client ET son ip print "Client distant connecte avec ip :"+str(ipDistante) # affiche IP du client #--- requete client --- requete=serverHTTP.readDataFrom(clientDistant) # lit les donnees en provenance client d'un coup #print requete # affiche requete recue - debug # analyse de la requete #====== si requete ajax ====== if requete.startswith("GET /ajax"): # si la requete recue est une requete ajax lignesRequete=requete.splitlines() # recupere la requete est list de lignes print lignesRequete[0] # premiere ligne = la requete utile #--- reponse serveur requete formulaire --- reponse=( # ( ... ) pour permettre multiligne.. httpResponse() # entete http OK 200 automatique fournie par la librairie Pyduino # contenu reponse AJAX + reponseAJAXServeur() # voir la fonction separee - pour clarte du code +"\n") # fin reponse serverHTTP.writeDataTo(clientDistant, reponse) # envoie donnees vers client d'un coup print "Reponse envoyee au client distant : " #print (bytes(reponse)) print (reponse) # affiche la reponse envoyee #====== si requete GET simple = premiere requete => envoi page HTML+JS initiale ====== elif requete.startswith("GET"): # si la requete commence par GET seul = premiere page print "Requete GET recue valide" #-- code Pyduino a executer au besoin global compt compt=0 # RAZ compt #--- reponse serveur requete initiale --- reponse=( # ( ... ) pour permettre multiligne.. httpResponse() # entete http OK 200 automatique fournie par la librairie Pyduino # contenu page HTML+ JS initiale + pageInitialeHTMLJS() # voir la fonction separee - pour clarte du code +"\n") # fin reponse serverHTTP.writeDataTo(clientDistant, reponse) # envoie donnees vers client d'un coup print "Reponse envoyee au client distant : " #print (bytes(reponse)) #print (reponse) # affiche la reponse envoyee - debug #====== si requete pas valide ====== else : # sinon requete pas valide print ("Requete pas valide") #====== une fois la page envoyée ====== #serverHTTP.close() # remarque : le socket = serveur doit rester ouvert # quand on quitte l'application: la connexion TCP reste active un peu donc erreur si re-execution trop rapide du code # on peut utiliser un port voisin dans ce cas... delay(50) # entre 2 loop() # -- fin loop -- #========== fonction fournissant la page HTML + JS initiale incluant code javascript AJAX ====== def pageInitialeHTMLJS(): # code Python a executer avant envoyer page # donnees types global data nombreValeurs=25; data=np.zeros((nombreValeurs,2)) # tableau de 1000 lignes x 2 colonnes data[:,0] = np.arange(0,nombreValeurs,1) # remplissage 1ere colonne = les x data[:,1] = np.arange(0,nombreValeurs,1) # remplissage 2eme colonne = les y #data[:,1] = np.random.normal(0,1,size=nombreValeurs) # remplissage 2eme colonne = les y1 # on laisse à 0 les 10ères valeurs #dataStr=",".join(data ) # pour convertir list en str avec elements separes par , dataGraph="" # initialise chaine dataStr=str(data.tolist()) # convertit tableau numpy pour affichage dygraph ++ dataGraph=dataGraph+dataStr # Note : lors envoi page initiale : donnees envoyees au sein code JS # lors envoi reponse ajax : donnees envoyees en tant que chaine texte # l'envoi initial d'un tableau de valeurs permet de fixer la largeur du graphique # options du graphique Dygraph a utiliser optionsGraph=""" labels: [ "x", "y="], // labels series //width : 800, // largeur // height: 400, // hauteur valueRange: [0,4095], // plage valeurs Y showRangeSelector: true // affiche l'outil de selection plage voulue """ contenuPageInitialeHTMLJS=( # debut page HTML """ Test reponse Ajax

Valeur courante =
Serveur Pyduino : Affichage Dygraphs d'une mesure analogique par reponse de requete Ajax
""" ) # fin page HTML+JS initiale return contenuPageInitialeHTMLJS # la fonction renvoie la page HTML #===================== Reponse AJAX ================== #--- fonction fournissant la page de reponse AJAX def reponseAJAXServeur(): # definition des variables a uiliser dans la reponse global data # operations sur tableau de donnees #data[:,1]=data[:,1]*2 # mesure la voie analogique A2 value=analogRead(A2) print "Value=" + str(value) #data[:,1]= value=analogRead(A2) data[:,1]=np.roll(data[:,1],-1) #fait defiler y de 1 cran vers la gauche data[-1,1]=value # modifie la valeur dernier elemnt Y # envoi du tableau en reponse Ajax #dataGraph="" #dataStr=str(data.tolist()) # convertit tableau numpy pour affichage dygraph ++ #dataGraph=dataGraph+dataStr # attention : envoyer une chaine car la fonction JS recoit une chaine : c'est pas du code JS qu'on envoie # donnees sous forme d'une chaine # format "x,y\n" dataGraph="" # attention - ici le formatage est la chaîne brute, pas la chaîne au sein de code JS.. subtil.. ! for i in range(len(data)-1): # format "x,y\n" dataGraph=dataGraph+str(data[i,0])+","+str(data[i,1]) + "\n" # le 2eme \n est dans la page HTML, le premier dans la chaine des donnees dataGraph=dataGraph+str(data[-1,0])+","+str(data[-1,1]) + "\n" # derniere ligne sans le plus print dataGraph # la reponse reponseAjax=( # debut page reponse AJAX #"""123, 456, 789""" dataGraph ) # fin page reponse AJAX return reponseAjax# la fonction renvoie la page HTML #--- obligatoire pour lancement du code -- if __name__=="__main__": # pour rendre le code executable setup() # appelle la fonction setup while not noLoop: loop() # appelle fonction loop sans fin