Utiliser une carte Arduino dans une station météo

Utiliser une carte Arduino
dans une station météo
Alain Jouve, Professeur de physique au Lycée Rosa Parks, Neuville sur Saône
Publié par Gérard Vidal et Charles-Henri Eyraud
Utiliser une carte Arduino dans une station météo
par Alain Jouve, Professeur de physique au Lycée Rosa Parks, Neuville sur Saône, Gérard Vidal, et Charles-Henri
Eyraud
Date de publication 2015-05-08
Résumé
Utiliser une carte Arduino dans une station météo
Table des matières
Utiliser une carte Arduino dans une station météo ............................................................... 1
Introduction .......................................................................................................... 1
Présentation de la station ........................................................................................ 1
Choix de la carte Arduino ....................................................................................... 3
Les cartes Arduino ......................................................................................... 3
Les contraintes .............................................................................................. 4
Le dispositif Arduino choisi ............................................................................ 4
Le module de communication sans fil ....................................................................... 5
Les capteurs ......................................................................................................... 7
Généralités ................................................................................................... 7
Le capteur de température et le capteur de pression ............................................. 8
L'hygromètre ................................................................................................. 9
L'anémomètre ................................................................................................ 9
L'abri ................................................................................................................. 11
Le boîtier émetteur ............................................................................................... 13
Alimentation de la carte Arduino et des capteurs ....................................................... 14
Le sketch à téléverser dans la carte Arduino ............................................................. 16
L'EDI Arduino ............................................................................................ 16
Structure générale d'un programme Arduino ..................................................... 16
Le sketch Arduino ....................................................................................... 16
Initialisation de la station ...................................................................................... 17
Le programme Python .......................................................................................... 17
Relevé et enregistrement des mesures ...................................................................... 18
Repérage des dates ....................................................................................... 18
Enregistrement des données ........................................................................... 19
Exemple de mesures enregistrée ..................................................................... 19
Annexes ............................................................................................................. 20
Le sketch Arduino ....................................................................................... 20
Le programme Python complet ....................................................................... 29
iii
Liste des illustrations
1. Vue de la station météo en état de fonctionnement ........................................................... 2
2. Synoptique du système complet .................................................................................... 3
3. La carte Arduino ....................................................................................................... 3
4. Le dispositif électronique de la station météo .................................................................. 4
5. Module radio XBee® "XBP24-AUI-001" ....................................................................... 5
6. XCTU : Copies d'écran de la page des paramètres du XBee ............................................... 5
7. Modification du paramètre SM SleepMode ..................................................................... 6
8. Pont diviseur de tension .............................................................................................. 6
9. Les capteurs fixés sur l'assiette inférieure de l'abri météo .................................................. 8
10. Capteur anénomètre ................................................................................................ 10
11. L'abri de la station météo ......................................................................................... 12
12. L'assiette inférieure ammovible ................................................................................. 13
13. Le boîtier émetteur : monté sur le mât support , intérieur du couvercle , partie fixe avec le
RIT et les connecteurs .................................................................................................. 14
14. Capteur photovoltaïque ............................................................................................ 15
15. Interface graphique du programme Python .................................................................. 18
iv
Utiliser une carte Arduino dans une
station météo
Alain Jouve, Professeur de physique au
Lycée Rosa Parks, Neuville sur Saône
Introduction
Les cartes Arduino sont des cartes "opensource" comportant un circuit microcontrôleur permettant
de réaliser des acquisitions de données issus de capteurs. Elles sont peu coûteuses et assez faciles
à mettre en oeuvre avec quelques lignes de codes de programmation. De nombreux sites proposent
des programmes divers et variés ce qui facilite la tâche des développeurs. Le site Arduino [http://
www.arduino.cc/fr/] est très complet .
Ces cartes peuvent gérer une communication sans fil type WiFi ou Bluetooth ce qui est très intéressant
dans le cas d'une station météo, il n'est pas nécessaire de se rendre sur le site pour acquérir les mesures.
Les modules XBee sont également des modules de télécommunication sans fil aisés à mettre en oeuvre.
Présentation de la station
La station météo se présente telle que le montre la photo ci-dessous.
1
Utiliser une carte Arduino
dans une station météo
Figure 1. Vue de la station météo en état de fonctionnement
Sur un pied de parasol de 14 kg est fixée une tige métallique normalement destinée à supporter un
antenne parabolique.Sur cette tige sont fixés le panneau solaire, l'abri ventilé(blanc) et le boîtier(gris)
contenant le montage electronique.
Dans l'abri ventilé se situent trois capteurs :
-un capteur de température
2
Utiliser une carte Arduino
dans une station météo
-un capteur de pression
- un capteur d'humidité
Un anémomètre à ailettes surmonte l'abri.
Un boitier protège l'ensemble du montage électronique qui lit et télétransmet les données issues des
capteurs. Une petite antenne le surmonte.
La station est énergétiquement autonome et le panneau solaire recharge une batterie de 12 V 12
Ah située dans une boîte étanche placée au pied de la station. Cette batterie alimente le montage
électronique et les capteurs.
Un logiciel implanté dans un ordinateur distant permet de récupérer périodiquement les données. La
structure générale de l'ensemble est comme l'indique la figure ci-dessous :
Figure 2. Synoptique du système complet
Choix de la carte Arduino
Les cartes Arduino
Les cartes Arduino sont bâties autour d'un microcontrôleur ATméga .
Il existe plusieurs modèle ayant chacun leurs spécificités. La plus courante est la carte Arduino Uno
qui a le mérite d'être robuste, et le microcontroleur, monté sur un support, peut être changé en cas
de panne. La carte Leonardo présente l'avantage de gérer deux ports série en même temps, mais le
microcontrôleur est soudé et ne peut pas être changé.
Figure 3. La carte Arduino
Ces cartes comportent toutes plusieurs entrées analogiques, des entrées et des sorties numériques. Le
convertisseur analogique numérique possède 10 bits et sa tension de pleine échelle est par défaut de
5 V mais peut être règlée entre 2,3 et 5 V.
On peut ajouter des fonctions grâce à des cartes aditionnelles appelées Shield qui se branchent
exactement dans les connecteurs de la carte Arduino. On peut superposer ainsi plusieurs Shield
3
Utiliser une carte Arduino
dans une station météo
Les contraintes
L'autonomie en énergie impose la consommation d'un faible courant.
Le panneau solaire photovoltaïque doit pouvoir recharger la batterie le jour, même l'hiver.
Les cartes Arduino comportent une mémoire vive de faible capacité : 32 ko pour l'UNO et 28 k pour
la carte Leonardo.
Les mesures doivent être assez rapprochées dans le temps pour un suivi précis de l'évolution rapide de
la météo locale lors d'orages par exemple. Les données doivent pouvoir être stockées dans la station,
puis ensuite téléchargées sur un ordinateur distant.
Il faut téléverser dans l'Arduino plusieurs bibliothèques qui occupent beaucoup de place mémoire,
la bibliothèque du port SPI qui gère la carte micro-SD, la bibliothèque du port I2C qui gère la
communication avec les capteurs.
Le dispositif Arduino choisi
Le dispositif comporte une carte Arduino UNO S3 surmontée d'un Shield "Carte micro-SD - Xbee".
Figure 4. Le dispositif électronique de la station météo
Le Shield comporte un lecteur de carte mirco-SD de 4 Go, ce qui permet d'enregistrer les données dans
un fichier de type texte. Le shield comporte également le connecteur pour Xbee.
4
Utiliser une carte Arduino
dans une station météo
Le module possède un petit interrupteur qui doit être placé sur la position "usb" lorsqu'on téléverse le
programme dans la carte Arduino. Lorsque le téléversement est terminé il faut placer l'interrupteur sur
la position "micro" afin que le module Xbee soit mis en service.
Le module de communication sans fil
Le module est directement fixé sur la platine Arduino soit via un module shield pour la carte Uno soit
directement sur la carte leonardo DFRobot qui possède un connecteur XBee. Il s'agit d'un module Xbee
pro série 1 avec un connecteur"UFL" qui permet de brancher une antenne via un fil de connection.
La fréquence d'émission est de 2,4 GHz.
Figure 5. Module radio XBee® "XBP24-AUI-001"
Une antenne extérieure est indipensable pour que la portée soit d'une centaine de mètre, car l'ensemble
est enfermé dans une boîte en matière plastique. Un autre module Xbee est connecté sur l'ordinateur via
un connecteur USB. Il permet au programme Python de communiquer avec la station météo.Ce dernier
module possède une antenne intégrée et n'est pas enfermé donc ne nécessite pas d'antenne aditionnelle.
Il faut paramétrer le XBee avec le logiciel XCTU qui est gratuit. Le paramétrtage usine convient
sauf pour la mise en veille. Pour que la mise en veille soit commandée par le logiciel il faut fixer le
paramètre "SM SleepMode" à la valeur de "Pin Hybernate1":
Figure 6. XCTU : Copies d'écran de la page des paramètres du XBee
5
Utiliser une carte Arduino
dans une station météo
Figure 7. Modification du paramètre SM SleepMode
Pour provoquer la mise en veille ou le réveil du XBee, le logiciel implanté dans l'Arduino place une des
sorties logiques (la sortie 9 par exemple) à l'état "HIGH" (mise en veille) ou l'état "LOW" (réveil). Il
faut réaliser une connection électrique avec un fil entre la sortie 9 de l'Arduino et la patte 9 du module
Xbee. ATTENTION : le module XBee n'accepte pas de tension supérieure à 3,3 V, il faut donc que
la borne 9 de l'arduino soit branchée sur un pont diviseur qui abaisse la tension de 4,8 V(état haut des
sortie Arduino) à 3 V. Le Xbee grille à la longue si on applique la tension de 4,8 V sur une de ses
pattes(votre serviteur en a fait l'amère expérience).
Figure 8. Pont diviseur de tension
Ce pont diviseur sera câblé sur le circuit imprimé qui supporte la carte Arduino. Le fil de liaison entre
la sortie du pont diviseur et la patte 9 du Xbee sera soudé sur la connexion du shield qui correspond
à la patte 9.
6
Utiliser une carte Arduino
dans une station météo
Les capteurs
Généralités
Trois des capteurs utilisés ici sont des capteur électroniques de la marque Sparkfun [https://
www.sparkfun.com/] . Ils utilisent le protocole I2C. Ils ont été acheté chez Lextronic [http://
www.lextronic.fr] et le site de Sparkfun permet de télécharger la librairie Arduino de chaque capteur
avec un exemple d'utilisation.
7
Utiliser une carte Arduino
dans une station météo
Le capteur de température et le capteur de pression
Figure 9. Les capteurs fixés sur l'assiette inférieure de l'abri météo
C'est le module BMP180 qui comporte un capteur de température et un capteur de pression. Il faut
d'abord lire la température qui est ensuite utilisée pour lire et établir la pression. On peut corriger la
valeur de la pression en la ramenant à l'altitude du niveau de la mer. La liaison avec l'Arduino s'effectue
à l'aide du protocole I2C.
8
Utiliser une carte Arduino
dans une station météo
L'hygromètre
Le module HTU21 de sparkfun mesure à la fois la température et l'hygrométrie. Il faut aussi utiliser
le protocole I2C.
L'anémomètre
C'est un anémomètre à ailettes acheté aussi chez Lextronic
9
Utiliser une carte Arduino
dans une station météo
Figure 10. Capteur anénomètre
Son principe est simple, un interrupteur de type ILS, commandé avec un petit aimant solidaire du
moulinet, se ferme à chaque tour. La carte Arduino compte le nombre de tours par seconde. Il faut
utiliser un petit montage dit "interrupeur sans rebond" entre la sortie de l'anémomètre et l'entrée
de l'arduino pour éliminer les oscillations amorties qui apparaissent parfois lors de la fermeture et
ouverture de l'interrupteur . Après étalonnage, avec un anémomètre du commerce, la vitesse du vent
est calculée par le logiciel de la carte Arduino.
10
Utiliser une carte Arduino
dans une station météo
L'abri
Pour que les mesures soient valables, il faut que les capteurs soient en contact direct avec l'air, mais
aussi protégés du soleil et de la pluie. L'abri doit donc être bien ventilé, mais suffisement protecteur
pour que les capteurs ne soient pas mouillés par le pluie . On utilise des assiettes de pot de fleur
en plastique renversées. L'idée provient du site : Soucabri [http://montpellier.meteo.free.fr/Soucabris/
Soucabri.htm]
11
Utiliser une carte Arduino
dans une station météo
Figure 11. L'abri de la station météo
Chaque assiette est séparée de sa voisine par une entretoise de 2 cm de long.Le rebord en creux de
chaque assiette est percé de trous afin d'éviter l'accumulation de l'eau de pluie ou de rosée.Les assiettes
sont peintes en blanc pour minimiser l'absorption de la lumière du Soleil qui réchaufferait l'abri et
rendrait erronée la valeur de la température.
L'assiette du bas porte les capteurs et est facilement amovible car fixée à l'aide d'écrous à ailettes.
12
Utiliser une carte Arduino
dans une station météo
Une connexion à l'aide d'un câble 4 fils permet de relier les capteurs avec l'Arduino avec le procole I2C.
Figure 12. L'assiette inférieure ammovible
Le boîtier émetteur
Il contient la carte Arduino et son Shield. L'antenne reliée au module XBee est fixée sur boîtier.
13
Utiliser une carte Arduino
dans une station météo
Figure 13. Le boîtier émetteur : monté sur le mât support , intérieur du
couvercle , partie fixe avec le RIT et les connecteurs
La carte Arduino est fixée sur une plaque imprimée sur laquelle sont fixés trois connecteurs :
1. Le connecteur d'alimentation
2. Le connecteur des capteurs utilisant la liaison I2C
3. Le connecteur relié à l'anémomètre
L'utilisation de connecteurs permet de débrancher facilement le boîtier émetteur et de l'emporter pour
une maintenance éventuelle, c'est pour cette même raison que l'antenne et l'ensemble Arduino sont
fixés dans le couvercle du boîtier que l'on peut facilement détacher du corps principal fixé sur le tube
support.
On trouve aussi sur la carte imprimée, le pont diviseur nécessaire à l'adaptation de la tension de la
borne 9 du Xbee, et un autre pont diviseur par 4 alimenté directement par la tension aux bornes de
la batterie. La sortie de ce pont diviseur est reliée à la borne A1 de l'Arduino. Le sketch lit la tension
appliquée sur cette borne et peut ainsi envoyer à l'ordinateur l'état de la charge de la batterie.
Le boîtier émetteur doit être étanche, mais doit cependant être ventilé pour éviter une surchauffe de
circuits électroniques, des trous sont percés en dessous du boîtier et dans la partie supérieure, ces
derniers sont recouverts d'une protection évitant que la pluie ne pénètre.
Alimentation de la carte Arduino et des
capteurs
Alimentation autonome avec une batterie de 12 V 12 Ah rechargée avec un panneau solaire de 0,15
m2. L'ensemble du dispositif consomme 0,06 A lorsque le module Xbee est en veille, et 0,12 A
lorsque le module Xbee est éveillé. L'éveil du Xbee dure 1min30 par 10 min. L'intensité moyenne
calculée sur heure est donc 0,07 A Donc l'autonomie de la batterie est de 12/0,07 = 171 h environ. Le
panneau solairephotovoltaïque polycristallin a pour puissance maximale Pm = 20 W, avec un courant
maximal de 1,2 A. Valeurs données pour une énergie recue par unité de surface de 4000 W/m2, c'est
à dire lorsque les rayons du Soleil sont orthogonaux à la surface du capteur, au mois de juin à Lyon.
14
Utiliser une carte Arduino
dans une station météo
L'expérence montre que la simple lumière du jour(même en hiver) suffit à compenser la consommation
de la station.
Figure 14. Capteur photovoltaïque
On peut éventuellement alimenter la station météo sous 12 V avec une petite ligne électrique issues
d'un transformateur situé à l'abri. La consommation de la carte Arduino et des capteurs étant très faible,
il n'y a pas de courant important (60 mA), et la chute de tension dans la ligne est réduite. Pour une
15
Utiliser une carte Arduino
dans une station météo
longueur de 100 m de câble 1 mm2, par exemple, la résistivité du laiton étant ρ = 7,0·10 -8 Ω.m , en
utilisant la relation :
(1)
on obtient une résistance de 14 Ω.(il faut tenir compte des deux fils) La chute de tension serait donc
de 0,8 V environ.
Le sketch à téléverser dans la carte Arduino
L'EDI Arduino
L'EDI "Arduino" est à télécharger gratuitement sur le site Arduino [http://www.arduino.cc/en/Main/
Software] dans l'onglet "download". On peut l'installer n'importe où sur son disque dur. Il comporte
de nombreuses bibliothèques. On tape le programme en langage C. Une fois terminé on le téléverse
dans la mémoire de la carte Arduino branchée sur le port USB de l'ordinateur.
Structure générale d'un programme Arduino
Un programme Arduino encore appelé sketch comporte toujours au moins deux procédures :
1. La procédure "setup" dans laquelle on fixe les valeurs des variables globales, on intialise le port
série utilisé et ses caractéristiques, etc ... Le setup est exécuté à chaque mise sous tension ou chaque
reset de la carte Arduino et uniquement.
2. La procédure "loop" qui ,comme son nom l'indique tourne en permanence, sur elle même tant que
l'Arduino est alimenté.
3. Des fonctions peuvent être définies également(facultatif mais recommandé) en dehors de ces deux
précédentes procédures , il faut les implémenter avec la procedure "loop".
Le sketch Arduino
Il est proposé dans les annexes.
Il comporte essentiellement :
1. L'importation des bibliothèques pour la gestion de la carte SD, du port I2C, des capteurs , et du
port SPI utilisé par la carte SD.
La création des variables : par exemple le fichier de sauvegarde des données, les repère temporels,
etc ...
2. Le setup dans lequel on ouvre le port série, et fixe les paramètres des entrées et sorties. Pour que
le lecteur de carte SD fonctionne correctement, il faut que la sortie 10 soie définie en sortie et soit
fixée à l'état haut. La borne 9 est définie en sortie et permet de mettre en veille ou de réveiller le
XBee. L'éveil est obtenu si elle est à l'état bas, et l'endormissement lorsqu'elle est à l'état haut.
On teste aussi l'a présence de la carte SD.
3. La boucle dans laquelle on trouve :
a) une série d'instructions uniquement réalisées lors de la première boucle :
lecture de la date courante sur l'ordinateur
16
Utiliser une carte Arduino
dans une station météo
test de l'existence et création du fichier de sauvegarde
b) Des instructions effectuées lors de chaque boucle
b1) la lecture du port série et l'exécution des instructions imposées par l'ordinateur :
1: Lecture des données
2: Lecture du fichier de sauvegarde de la date courante
3 : Lecture des fichiers antérieurs
b2) L'enregistrement des valeurs toute les minutes
b3) La mise en veille et le réveil du XBee
4. Des fonctions :
lecture de la tempéraure, de la pression, de l'hygrométrie, et de la vitesse du vent
Sauvegardes des données sur la carte SD
initialisation de la date et de l'heure
calcul de la date et de l'heure
lecture du Xbee
changement du fichier de sauvegarde
lecture de la date et de l'heure sur le module timer RTC(éventuellement)
Initialisation de la station
Il faut d'abord lancer le logiciel situé sur l'ordinateur qui permettera d'intialiser la date et l'heure.. Après
la mise sous tension de la carte Arduino, appuyer sur le bouton "reset". Le sketch efffectue d'abord
le "setup" puis la "loop". Lors de la première boucle il demande à l'ordinateur la date et l'heure en
envoyant le message "T". Le programme Python lit la date sur l'ordinateur et l'envoie à la station.
Le programme Python
Il est donné en annexe. On utilise la biliothèque Tkinter pour créer une interface graphique. La fenêtre
qui s'affiche comporte les éléments suivants :
1. Les valeurs des mesures qui s'affichent en temps réel, rafraîchissement toutes les 10 minutes, avec
une indication de l'heure de l'enregistrement de ces mesures.
2. La commande de détection du module Xbee connecté à l'ordinateur sur le port USB
3. La commande de lancement des mesures, lorsque l' heure est de la forme hh mm 00
4. La commande de désactivation de la mise en veille de la station en cas de mise au point ou de
maintenance.
5. La commande de lecture du fichier journalier en cours.
6. La commande de lecture de tous les fichiers enregistrés sur la carte SD de la station.
7. Une information sur l'état de veille ou non de la station, avec l'heure de la précédente mise en veille :
un voyant est rouge si la station est en veille et est vert sinon.
17
Utiliser une carte Arduino
dans une station météo
Figure 15. Interface graphique du programme Python
La charge de la batterie est affichée à titre informatif, mais mais n'est pas enregistrée.
La fermeture de la fenêtre et l'arrêt du programme sont obtenus en cliquant sur le petit carré muni
d'une croix rouge.
Deux Timers sont créés dans le programme: l'un de 1 h qui lit les valeurs à chaque heure(facultatif),
l'autre de 2 s qui scrute le port XBee et qui détecte s'il se réveille. Il commande la mesure et met l'heure
de l'Arduino à jour.
Relevé et enregistrement des mesures
Repérage des dates
Le module Arduino étant livré à lui-même il faut repérer les dates à l'aide de son horloge interne. Il
est possible d'utiliser l'instruction millis() (format type long) qui donne le nombre de milliseconde
écoulées depuis le "reset" de la carte.Le sketch calcule le nombre de secondes, minutes, heures, jours,
mois écoulés depuis le reset. L'horloge interne n'est pas stable et présente une importante dérive.
Plusieurs essais ont conduit à corriger le nombre de millisecondes en ajoutant 1250 ms par minute.
Cela n'est pas très précis.
18
Utiliser une carte Arduino
dans une station météo
On peut utiliser petit module un timer externe branché sur la carte Arduino qui est plus précis, mais
qui présente un problème de compatibilité avec la carte SD car il utilise le même port SPI avec un
paramétrage différent.
Chaque fois que la station envoie des données vers l'ordinateur, toutes les 10 minutes si l'ordinateur
structe le port XBee, elle demande la date et l'heure à l'ordinateur qu'elle corrige si besoin.
Enregistrement des données
Un module Xbee est relié au port USB de l'ordinateur. Un programme écrit en Python , ou autre
langage, scrute toutes les deux secondes les messages envoyés par la station. Lorsque le réveil a lieu,
le sketch Arduino envoie un message "up" ce qui déclenche une demande de relevé de mesures. Le
sketch Arduino relève les mesures et les envoie. Ces mesures sont enregistrées dans un fichier type
"csv" dans l'ordinateur. Lorsque la station se place en mode veille la station envoie le message"down"
ce qui indique au programme Python que la station est inaccessible.
Si l'ordinateur se met en veille, il n'est pas possible de relever les valeurs de la station météo. Le
sketch Arduino enregistre systématiquement(en veille ou non) sur la carte SD de la station les données
toutes les minutes dans un fichier type "csv" nommé "mm-jj-U.TXT". avec mm: mois, jj :jour. La
bibliothèque SD ne gère pas les fichiers dont le nom est différent de la structure ********.***(8
caractères simples+ '.' + ***)
Structure
d'une
ligne
enregistrée
dans
"aaaa;mm;jj;hh,mm,ss;tempe;pression;hygrométrie;vent;"
le
fichier
csv
:
Le sketch Arduino change automatiquement de fichier d'enregistrement chaque jour à 0 h 00. Il y a
donc un fichier par jour.
Le programme python comporte une commande permettant de lire le fichier en cours et une autre
commande permettant de lire tous les fichiers enregistrés. Les données sont alors stockées dans
l'ordinateur dans un fichier de type "csv" .
Exemple de mesures enregistrée
Date, heure température, pression atm, hygrométrie,vitesse du vent
27/10/2014,1h10,8.36,1025.84,101.38,0.00
27/10/2014,1h11,8.37,1025.89,101.41,0.00
27/10/2014,1h12,8.33,1025.79,101.26,0.00
27/10/2014,1h13,8.29,1025.72,101.23,0.00
27/10/2014,1h14,8.31,1025.76,101.28,0.00
27/10/2014,1h15,8.31,1025.82,101.35,0.00
27/10/2014,1h16,8.38,1025.82,101.50,0.00
27/10/2014,1h17,8.36,1025.80,101.41,0.00
27/10/2014,1h18,8.31,1025.82,101.25,0.00
27/10/2014,1h19,8.29,1025.78,101.28,0.00
27/10/2014,1h20,8.27,1025.80,101.25,0.00
27/10/2014,1h21,8.27,1025.83,101.22,0.00
27/10/2014,1h22,8.29,1025.81,101.36,0.00
27/10/2014,1h23,8.29,1025.95,101.35,0.00
27/10/2014,1h24,8.29,1025.86,101.40,0.00
27/10/2014,1h25,8.29,1025.85,101.40,0.00
27/10/2014,1h26,8.26,1025.87,101.42,0.00
27/10/2014,1h27,8.27,1025.86,101.40,0.00
27/10/2014,1h28,8.24,1025.86,101.35,0.00
27/10/2014,1h29,8.21,1025.85,101.30,0.00
27/10/2014,1h30,8.19,1025.84,101.26,0.00
27/10/2014,1h31,8.18,1025.94,101.23,0.00
27/10/2014,1h32,8.16,1025.88,101.19,0.00
19
Utiliser une carte Arduino
dans une station météo
27/10/2014,1h33,8.13,1025.81,101.17,0.00
27/10/2014,1h34,8.12,1025.90,101.14,0.00
27/10/2014,1h35,8.10,1025.86,101.14,0.00
27/10/2014,1h36,8.10,1025.93,101.15,0.00
27/10/2014,1h37,8.09,1025.86,101.17,0.00
27/10/2014,1h38,8.10,1025.83,101.20,0.00
27/10/2014,1h39,8.09,1025.87,101.18,0.00
27/10/2014,1h40,8.07,1025.79,101.17,0.00
27/10/2014,1h41,8.06,1025.78,101.16,0.00
27/10/2014,1h42,8.04,1025.78,101.15,0.00
27/10/2014,1h43,8.04,1025.81,101.15,0.00
27/10/2014,1h44,8.03,1025.76,101.15,0.00
27/10/2014,1h45,8.02,1025.77,101.11,0.00
27/10/2014,1h46,7.98,1025.79,101.03,0.00
27/10/2014,1h47,7.94,1025.78,100.96,0.00
27/10/2014,1h48,7.90,1025.78,100.90,0.00
Annexes
Le sketch Arduino
#include < SPI.h>//ouvre le port qui gère la carte SD
/* SD card attached to SPI bus as follows:
** MOSI - pin 11
** MISO - pin 12
** CLK - pin 13
** CS - pin 4*/
#include < SD.h>//carte SD
#include "HTU21D.h"//bibliothèques des capteurs sparkfun humidité
#include SFE_BMP180.h// bibliothèques des capteurs sparkfun pression
#include Wire.h//Pour le port I2C
File Myfile;
char fichier_[12];
SFE_BMP180 pressure;//Capteur de pression et de température
HTU21D myHumidity;// capteur d'hygrométrie
long minute_debut;
char commande_mesure;
char message;
int mois_;
int jour_;
20
Utiliser une carte Arduino
dans une station météo
int an_;
int heure_=0;
int minute_=0;
int seconde_ = 0;
int anemometre = 2; //Anemometre connecte sur le port 2
long date_ini_veille;
boolean debut=true;//A la mise sous tension de l'Arduino, debut = true afin que la boucle loop effectue l'initialisation de
long date_ini,date_depart,date_depart_;
//long date_courante;
boolean Xbee_reveille=true;
//String reponse="";
char status;
long temps;
void setup()
{
pinMode(anemometre, INPUT);
Serial.begin(9600); //Port Xbee
pressure.begin();
delay(10);
myHumidity.begin();
delay(100);
//Réglage en mode veille du Xbee;
// La borne 9 de l'Arduino est reliée à la borne9 du Xbee
date_depart=millis();
debut=true;
pinMode(9, OUTPUT);//borne de sortie pour commander la mise ne veille du Xbee nécéssite le réglage de avec XCT
digitalWrite(9, LOW);//mode réveillé du XBee
pinMode(10, OUTPUT);
digitalWrite(10, HIGH);//obligatoire pour que la carte sd soit reconnue
//SD.begin(4);//Sur leonardo ou mega voir https://learn.adafruit.com/adafruit-data-logger-shield/for-the-mega-and-leo
if (SD.begin(4)){//vérification de la présence de la carte SD
Serial.println("La carte SD est reconnue");
delay(1000);
return;
}
else{
Serial.println("La carte SD n'est pas reconnue");
}
}
void loop(){
//double T,P,p0;
if (debut==true){
initialisation_jour_heure();
21
Utiliser une carte Arduino
dans une station météo
changt_fichier();
delay(1000);
if (!SD.exists(fichier_)){
Myfile = SD.open(fichier_, FILE_WRITE);
Myfile.println("date ; heure; température; pression; hygrométrie;vent :");
Myfile.close();
}
else{
}
delay(1000);
}
date_ini=millis();
debut =false;
Serial.write("down");
delay(100);
digitalWrite(9,HIGH);
date_ini_veille=millis();
Xbee_reveille = false;
if (Serial.available()){//Si le port Xbee recoit une donnée de la part de l'ordinateur par les ondes wifi
commande_mesure=Serial.read();//Lecture de la valeur sur le port Xbee
delay(100);
if (commande_mesure==49){//Si la valeur est de 49; valeur 1 en code ASCII envoyé par processing : mesures instan
delay(1000);
//envoie des quatres valeurs les unes après les autres, cela se traduit par un chaîne de caractère recue par le program
Serial.print(mesure_temperature());//envoie sur le Xbee
delay(400);//delais technique
Serial.print(';');//séparation des données
delay(10);
Serial.print(mesure_pression(mesure_temperature()));
delay(400);
Serial.print(';');
delay(10);
Serial.print(mesure_hygrometrie());
delay(400);
Serial.print(';');
delay(10);
Serial.print(vitesse_vent());
delay(400);
Serial.print(';');
delay(10);
Serial.print(analogRead(0));//Envoyer surle port Xbee la valeur du niveau de la batterie
delay(10);
delay(3000);//délais technique
22
Utiliser une carte Arduino
dans une station météo
initialisation_jour_heure();
delay(2000);
date_ini=millis();
Serial.flush();
}
if (commande_mesure==50){//Si la valeur est de 50; valeur 2 en code ASCII envoyé par processinglectures des donn
if (SD.exists(fichier_)){
//Serial.print(fichier_);
Myfile = SD.open(fichier_);
while(Myfile.available()){
Serial.print( char(Myfile.read()));
}}
}
delay(10);
Myfile.close();
if (commande_mesure==51){ // Si la valeur est 3 : valeur 51 en code ASCII: Lecture des données des fichiers non e
delay(100);
File racine = SD.open("/");
racine.rewindDirectory();
while(true) {
File fichier_donnees = racine.openNextFile();
char fichier_tmp[12];
if (! fichier_donnees) {
// no more files
//Serial.println("**nofiles**");
//racine.rewindDirectory();
break;
}
Serial.println(fichier_donnees.name());
if (fichier_donnees.name()[6]=='U'){
//boolean transfere = false;
for(int n = 0;n <12;n++){
fichier_tmp[n]=fichier_donnees.name()[n];
}
fichier_donnees.close();
//Serial.println(fichier_tmp);
Myfile = SD.open(fichier_tmp);
while(Myfile.available()){
Serial.print( char(Myfile.read()));
}
/*transfere=true;
delay(10);
if (transfere==true){
char fichier_tmp_[12];
for(int n = 0;n 12;n++){
fichier_tmp_[n]=fichier_tmp[n];
}
fichier_tmp_[6]='L';
File Myfile_tmp = SD.open(fichier_tmp_,FILE_WRITE);
23
Utiliser une carte Arduino
dans une station météo
Myfile.seek(0);
while(Myfile.available()){
Myfile_tmp.write(Myfile.read());
}
Myfile_tmp.close();
}*/
Myfile.close();
}
else{fichier_donnees.close();}
}
//racine.close();
}
}
/**********************************************************************************************
*
MESURES toutes les 60 s- mise en veille
*
***********************************************************************************************
double tempe_;
double pression_;
float hygro_;
float vent_;
if (millis()>=date_ini + 60000){//mesure effectuée environ toutes les minutes
digitalWrite(9,LOW);//réveil du Xbee
delay(1000);//délais technique dû au temps de réveil du Xbee
tempe_ =mesure_temperature();
delay(400);
pression_= mesure_pression(tempe_);
delay(400);
hygro_=mesure_hygrometrie();
delay(400);
vent_=vitesse_vent();
delay(1100);
determination_date();//Calcul de la date de la mesure
sauv_donnees(tempe_,pression_,hygro_,vent_);//sauvegarde dans le fichier du jour des dates et valeurs
date_ini=millis();//initialisation de la date de la mesure en milliseconde
digitalWrite(9,HIGH);
}
if ((millis()=date_ini_veille + 600000)(Xbee_reveille == false)){//réveil du Xbbe toutes les 10 min pendant 1 min pou
digitalWrite(9,LOW);
delay(100);
Serial.write("up");
Xbee_reveille = true;
}
if ((millis()>=date_ini_veille + 690000)(Xbee_reveille == true)){//réveil du Xbbe toutes les 10 min pour permettre la
Serial.write("down");
delay(100);
digitalWrite(9,HIGH);
date_ini_veille=millis();
Xbee_reveille = false;
24
Utiliser une carte Arduino
dans une station météo
}
//Serial.flush();
}
/**********************************************************************************************
*
Diverses fonctions
*
***********************************************************************************************
double mesure_temperature(){// fontion de mesure de la température codée sur un entier
double T;
status = pressure.startTemperature();//la mesure de la température s'effectue sur le module sparkfun BMP201 et utilisa
if (status != 0)//si le module répond
{
delay(status);//délais imposé par le module
// Retrieve the completed temperature measurement:
// Note that the measurement is stored in the variable T.
// Function returns 1 if successful, 0 if failure.
status = pressure.getTemperature(T);
}//lecture de la température
return T;//renvoie la valeur de T
}
double mesure_pression(double T){//fonction de mesure de la pression codée sur un entier
double P;//défintion de la variable pression
status = pressure.startPressure(3);//demarrage de la mesure de la pression
if (status != 0)//si le module répond
{
delay(status);//délais imposé par le module
status = pressure.getPressure(P,T);//lecture de la pression sur le module
delay(50);
}//imposé par le calcul qui suit
float p0 = pressure.sealevel(P,200);//Pression corrigée de l'altitude de 200 m, changer la veleur selon le lieu
delay(50);//imposé par le calcul précédent
return p0;//renvoie la pression corrigée de l'altitude
}
float mesure_hygrometrie(){//fonction de mesure de le l'hygrométrie codée sur un réel
float humd = myHumidity.readHumidity(); //lecture de la valeur de l'humidité sur le module sparkfun
return humd; //renvoie la valeur de l'humidité
}
float vitesse_vent(){//Mesure et Calcul la vitesse du vent : http://www.lextronic.fr/P28025-capteur-anenometre.html
unsigned long vent= 0;
vent=pulseIn(anemometre,HIGH,800000);
//Serial.println(vent);
//Serial.println(vent2);
float valeur=0;
if (vent>0){
valeur=(70000.0*6.2/vent)*3.6;
//Serial.print("valeur=");
//Serial.println(valeur);
}
//formule donnée par le constructeur du moulinet
//Serial.print(valeur);
}
return valeur;
25
Utiliser une carte Arduino
dans une station météo
String reponse_Xbee(){//fonction de lecture des donnees recues par le Xbee lors
int n=0;
while(!Serial.available()){
n++;
if(n>50000){break;}};
char byte_recu = Serial.read();//le xbee envoie un succession de charactère
String reponse_="";
while (byte_recu!=-1){ //si on lit -1 sur le port série cela signifie que la communication n'a pas lieu et on répète tant q
}
if (byte_recu!=-1){//on vérifie de nouveau car la boucle while comporte deux lectures et la deuxième peut être mauv
reponse_=reponse_+byte_recu;
}//on crée une chaîne de caractère en concaténant les charactères reçus
byte_recu = Serial.read();//lecture du charactère
delay(20);
} //délais technique
//Xbee.println(reponse_);
return reponse_ ; //renvoie de la chaîne réponse
//Xbee.flush();
void sauv_donnees(float tempe_,float pression_, float hygro_,float vent_){//Sauvegarde des données dans le fichier de
if (SD.exists(fichier_)) {
Myfile = SD.open(fichier_, FILE_WRITE);
Myfile.print(an_);
Myfile.print(";");
Myfile.print(mois_);
Myfile.print(";");
Myfile.print(jour_);
Myfile.print(";");
Myfile.print(heure_);
Myfile.print(";");
Myfile.print(minute_);
Myfile.print(";");
Myfile.print(seconde_);
Myfile.print(";");
Myfile.print(tempe_);
Myfile.print(";");
Myfile.print(pression_);
Myfile.print(";");
Myfile.print(hygro_);
Myfile.print(";");
Myfile.println(vent_);
//Myfile.println(":");
//Myfile.print("\r\n");
delay(10);
Myfile.close();}
else {changt_fichier();}
}
void initialisation_jour_heure(){//fonction nécessaire si on ne dispose pas d'un module timing extérieur
String reponse="";
26
Utiliser une carte Arduino
dans une station météo
boolean reponse_=false;
//int n=0;
//Serial.println("initialisation de la date");
Serial.flush();//effacement des données sur tampon de réception du port Xbee
delay(1000);
Serial.print('T');
delay(2000);
reponse=reponse_Xbee();
delay(1000);
if (reponse!=""){
//Serial.println(reponse);
//Serial.println("date recue");
int repere_1= reponse.indexOf(';');
int repere_2= reponse.indexOf(';',repere_1+1);
//Serial.println(repere_1);
int repere_3 = reponse.indexOf(';',repere_2+1);
//Serial.println(repere_2);
int repere_4 = reponse.indexOf(';',repere_3+1);
//Serial.println(repere_3);
int repere_5 = reponse.indexOf(';',repere_4+1);
//Serial.println(repere_4);
if((repere_1!=0)(repere_2!=0)(repere_3!=0)(repere_4!=0)(repere_5!=0)){
an_=reponse.substring(0,repere_1).toInt();
mois_=reponse.substring(repere_1+1,repere_2).toInt();
// Serial.println(mois_ini);
jour_=reponse.substring(repere_2+1,repere_3).toInt();
// Serial.println(jour_);
heure_=reponse.substring(repere_3+1,repere_4).toInt();
//Serial.println(heure_);
minute_=reponse.substring(repere_4+1,repere_5).toInt();
//Serial.println(minute_);
seconde_=reponse.substring(repere_5+1).toInt();
//Xbee.println(minute_);
//changt_fichier();
//delay(1000);
date_ini=seconde_*1000;}}
Serial.flush();
}
void determination_date(){//fonction nécessaire si on ne dispose pas d'un module timing extérieur
boolean changt_date = false;
long date_presente = millis();
long temps_ecoule=date_presente-date_ini+1250;
temps_ecoule=temps_ecoule/1000;
if ((seconde_+temps_ecoule)59){seconde_ = seconde_+temps_ecoule;}
else{seconde_=(temps_ecoule-59);
if (minute_<59){minute_=minute_+1;}
else{minute_=0;
27
Utiliser une carte Arduino
dans une station météo
if(heure_<23){heure_=heure_+1;}
else{heure_=0;
switch (mois_) {
case 1 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 2;}
break;
case 2 :if (jour_<28){jour_=jour_+1;}
else{jour_ = 1;
mois_= 3;}
break;
case 3 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 4;}
break;
case 4 :if (jour_<30){jour_=jour_+1;}
else{jour_ = 1;
mois_= 5;}
break;
case 5 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 6;}
break;
case 6 :if (jour_<30){jour_=jour_+1;}
else{jour_ = 1;
mois_= 7;}
break;
case 7 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 8;}
break;
case 8 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 9;}
break;
case 9 :if (jour_<30){jour_=jour_+1;}
else{jour_ = 1;
mois_= 10;}
break;
case 10 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 11;}
break;
case 11 :if (jour_<30){jour_=jour_+1;}
else{jour_ = 1;
mois_= 12;}
break;
case 12 :if (jour_<31){jour_=jour_+1;}
else{jour_ = 1;
mois_= 1;
an_= an_+1;}
break;
}
28
Utiliser une carte Arduino
dans une station météo
changt_fichier();
delay(1000);}}
}
}
void changt_fichier(){
int n=0;
String string_mois;
String string_jour;
if (mois_<10){string_mois = "0"+ String(mois_);}
if (mois_>=10){string_mois = String(mois_);}
if (jour_<10){string_jour = "0"+ String(jour_);}
if (jour_>=10){string_jour = String(jour_);}
String nom_fichier=string_mois+"-"+string_jour+"-"+"U"+"."+"TXT";
while(n<11){
fichier_[n]=nom_fichier[n];
n++;
}
//Serial.println("nom du fichier ="+nom_fichier);
if (!SD.exists(fichier_)){
Myfile = SD.open(fichier_, FILE_WRITE);//Creation du fichier de sauvegarde
Myfile.println("date ; heure; temperature; pression; hygrometrie;vent :");//Premiere ligne de la sauvegarde
Myfile.close();//Fermeture du fichier
}
}
]]
Le programme Python complet
# -*- coding: utf-8 -*#
# Module de lecture/ecriture du port Xbee station météo
from Tkinter import *
from serial import *
import time
import threading
from datetime import datetime,date,timedelta
from time import gmtime, strftime
import platform
from serial.tools import list_ports
import os, io,itertools
import signal
import sys
import pylab
from PIL import Image, ImageTk
import tkMessageBox
if platform.system() == 'Windows':#Détection du système , utilisation du registre windows
29
Utiliser une carte Arduino
dans une station météo
else:
import _winreg as winreg
import glob
port_Xbee=''
path = 'HARDWARE\\DEVICEMAP\\SERIALCOMM'
marche=True
debut=True
forcage=False
mise_en_veille_active=True
def enumerate_serial_ports(): # fonction listant l'ensemble des ports série(USB) utilisés sur la machine
#définition du chemin de la clef du registre donnant la lsite des ports séries en activité
try:
key = winreg.OpenKey(winreg.HKEY_LOCAL_MACHINE, path) #défintion de la clef du registre des po
except WindowsError:
raise Exception
for i in itertools.count():
#pour l'ensemble des ports série détectés,les lister dans l'ordre croissan
try:
val = winreg.EnumValue(key, i)
# valeur des différents ports
yield (str(val[1])) # , str(val[0])) #La fonction yield permet de garder la liste des ports classés
#print(str(val[i]))
except EnvironmentError:
print("erreur")
break
def detection_port_Xbee():
global Xbee
liste_des_ports=enumerate_serial_ports()
for i in liste_des_ports :#Détetection du port série(USB) sur lequel est branché le module xbee
print(i)
try :
with Serial(port=i, baudrate=9600, timeout=1, writeTimeout=1) as Xbee:
if Xbee.isOpen():
time.sleep(1)
Xbee.write('+++')
time.sleep(1.2)
ligne = Xbee.readline()
if (ligne=='OK'+chr(13)) :
port_Xbee=i
#return port_Xbee
break
Xbee.close()
except :
continue
Xbee = Serial(port=port_Xbee, baudrate=9600, timeout=1, writeTimeout=1)
chaine = "Xbee branché sur le port :" +port_Xbee
B1.config(state=DISABLED)#change l état des boutons
B3.config(state=NORMAL)
30
Utiliser une carte Arduino
dans une station météo
#
B5.config(state=NORMAL)
text1.set(chaine ) #indication du port utilisé
detection_creation_fichier_sauvegarde() #Détecte si le fichier de sauvegarde est présent
def Configuration_Xbee() :#Entrée dans le mode configuration du Xbee relié à l'orinateur (pas utilisé)
while Xbee.isOpen():
time.sleep(1)
Xbee.write('+++')
#activation du mode config du Xbee
time.sleep(1.2)
ligne = Xbee.readline()
if (ligne=='OK'+chr(13)) :
#Vérification de la réponse du Xbee
print(ligne)
print( 'mode config')
Xbee.write('ATID'+chr(13)) #lecture de n° identification du Xbee
time.sleep(1)
ligne = Xbee.readline()
time.sleep(0.2)
print(ligne)
Xbee.write('ATCN'+chr(13)) # on quitte le mode config
time.sleep(1)
ligne = Xbee.readline()
time.sleep(0.2)
print(ligne)
reponse =raw_input("Taper q pour quitter le mode config :")# Demande à l'opérateur si on doit quitter
#le mode config la fonction input n'est pas adaptée en python 2 :
#interprétation du code
print reponse
if (reponse == 'q'):
break
print('terminé')
#Envoie de données de la carte Arduino vers l'ordinateur:
def lecture_des_donnees(moment_d_acquisition) :#Commande à l'arduino de lire les données
if (Xbee.isOpen()) :
Xbee.write('1') #code 1 : indique à l'arduino qu'il faut lire les données
b.stop()#arrêt du timer de 10 s pour eviter les conflits de lecture du port série
time.sleep(2)#pause de 2s
mesure_ = Xbee.readline()#lecture du port série de la ligne envoyé par l'Arduino
time.sleep(2)#mise en veille technique
mesure=mesure_.split(";")#Sépare les données et les transforme en tableau : le';' est envoyépar l'arduino en
print mesure#écrit la ligne dans la console pour vérification
if (len(mesure)==5):#si il y a 5 éléments dans le tableau la lecture est sans doute bonne
temperature=mesure[0] #l'élément 1 du tableau est la tempé
#print('Temperature = '+temperature)
text_tempe.set( "Tempé =" +temperature[:4] + "°C")
31
Utiliser une carte Arduino
dans une station météo
pression=mesure[1] #l'élément 2 du tableau est la pression
#print('Pression = '+ pression)
text_pression.set("Pression = "+pression[:4] + " hPa")
hygro = mesure[2]#l'élément 3 du tableau est l'hygrométrie
#print('Hygrométrie =' +hygro)
text_hygro.set("Hygrométrie = " + hygro[:4]+" % hum")
vent=mesure[3]#l'élément 4 du tableau est la vitesse du vent
text_vent.set("vent ="+vent[:4]+"km/h")
batterie = mesure[4]#l'élément 5 du tableau est la tension aux bornes de la batterie
#print('Batterie =' +batterie)
y=int(float(batterie)/860*700)#donne un nombre de pixel proportionnel à la tension de charge de la batte
tension=round(float(batterie)/1024.0*16.0,1)#Permet d'obtenir la tension à partir du nombre binaire
texte=str(tension)+ " V"
#print(y)
text2.set("Mesures : Temperature = " + temperature +"°C ;" +" Pression = " +pression+"hPa ;"+" Hygro
text3.set("Heures des données affichées : "+ str(moment_d_acquisition.hour) +":"+ str(moment_d_acqui
can1.delete(ALL)#effacemement du canvas de charge de batterie
if (tension>=9.0) :
poly2=can1.create_polygon(0, 0, y, 0,y,50,0,50, fill='green')#Si la charge de la batterie est bonne on t
else :
poly2=can1.create_polygon(0, 0, y, 0,y,50,0,50, fill='red')#Si la charge de la batterie est insuffisant on
#tracé d'un rectangle de couleur et de longueur proprotionnelle à la tesnion aux bornes de la batterie
texte1=can1.create_text(y, 12, text=texte, font="Arial 12 italic", fill="black")
texte2=can1.create_text(350, 30, text="charge batterie", font="Arial 16", fill="blue")
return [temperature,pression,hygro,vent,batterie]#La fonction de lecture des données renvoie les valeurs
Xbee.flushInput()#efface les données du tampon de réception du port série
b.start();#redémarrage du timer de 2 s : lecture du port série
else :
can3.delete(ALL)
text2.set("La station météo n'est pas prête")#Si la lecture est mauvaise
can2.create_oval(5,5,45,45,fill='red')#le voyant rouge s'allume
can3.create_text(200,24, text =' Rouge : station en mode veille')#indique que la station ne répond pas
return [0,0,0,0]
Xbee.flushInput()#efface les données du tampon de réception du port série
b.start() #redémarrage du timer de 2 s : lecture du port série
else :
print('Le module Xbee ne répond pas')
return [0,0,0,0]
b.start();#redémarrage du timer de 2 s : lecture du port série
def sauvegarde_donnees(donnees) :#Sauvegarde des données dans un fichier csv
if os.path.isfile('C:\IFE\Météo\documents_contribution\programme python\donnees_meteo.csv') :
fichier = open('C:\IFE\Météo\documents_contribution\programme python\donnees_meteo.csv',"a")
#fichier.write( strftime("%a, %d %b %Y %H:%M", gmtime())+';')
for i in range(6) :
fichier.write(str(donnees[i]))
32
Utiliser une carte Arduino
dans une station météo
fichier.write(';')
fichier.write('\n')
fichier.close()
else :
print('Le fichier n''existe pas')
donnees=[0,0,0,0] #pour éviter une erreur d'écriture
def test_date_recuperation():
if os.path.isfile('C:\IFE\Météo\documents_contribution\programme python\donnees_meteo.csv') :
fichier = open('C:\IFE\Météo\documents_contribution\programme python\donnees_meteo.csv',"r")
chaine = fichier.read()
print chaine
fichier.close()
def detection_creation_fichier_sauvegarde(): #Vérification de l'existence d'un fichier de sauvegarde, sinon le créer
if (os.path.isfile('C:\IFE\Météo\documents_contribution\programme python\donnees_meteo.csv')!=True) :
fichier = open('C:\IFE\Météo\documents_contribution\programme python\donnees_meteo.csv',"w")
fichier.close()
def fermer_programme():
"""Fonction appelée quand vient l'heure de fermer notre programme"""
print("C'est l'heure de la fermeture !")
if (Xbee.isOpen()) :
Xbee.close()
if (a) :
a.stop()
if(b) :
b.stop()
fenetre_meteo.destroy()
def mise_enveille_Xbee(moment_d_acquisition):
if (Xbee.isOpen()) :
time.sleep(1)
Xbee.write('5') # 5 est le code de mise en veille du Xbee station
time.sleep(1)
Xbee.write(str(moment_d_acquisition.hour))#Donne l'heure à la station
time.sleep(1)
Xbee.write(str(moment_d_acquisition.minute))#Donne les minutes à la station
time.sleep(1)
def mesures() :
#fonction appelée par le timer toutes les 60s, la mesure n'est ici effectuée que chaque heure
global debut
global forcage
B2.config(state=DISABLED)#le bouton arret des mesures est désactivé
#if( ((moment_d_acquisition.minute%2)==0)and (moment_d_acquisition.second==0)):#toutes les heures (
moment_d_acquisition = datetime.now()
33
Utiliser une carte Arduino
dans une station météo
if (moment_d_acquisition.minute>0):#pour que l'acquisition n'ai lieu qu'a l'heure pile
global acquisition_effectuee#variable globale indiquant que l'acquisition est effectuée
acquisition_effectuee =False
if (((moment_d_acquisition.minute==0)and (acquisition_effectuee==False))or(debut==True)or(forcage==T
b.stop()#L'acquisition a lieu lorsque le timer de 2 s est à l'arrêt
time.sleep(1.5)#Mise en veille technique.
Xbee.flushInput() #effacement des données entrantes du port série
debut=False #indique que ce n'est plus la première mesure
forcage=False #indique que l'on de force pas la lecture de la meusre à un moment quelconque
acquisition_effectuee=True;
donnees=lecture_des_donnees(moment_d_acquisition) #appel de la lecture des données sur l'Arduino di
jour = datetime.now()
date_= str(jour.day)+"/"+str(jour.month)+"/"+str(jour.year)#lecture de la date et de l'heure
heure_=str(jour.hour)+"h"+str(jour.minute)
donnees.insert(0,heure_)
donnees.insert(0,date_)
print(donnees)
sauvegarde_donnees(donnees)#appel de la fonction de sauvgarde des données
#mise_enveille_Xbee(moment_d_acquisition)#met en veille le Xbee pour préserver les batteries de la st
Xbee.flushInput()
b.start()
B2.config(state=NORMAL)#le bouton arret des mesures est activé
def scrutation_mise_en_veille() :#fonction applée par le timer de 2 s pour srcuter la mise en veille de la station
examen_veille=Xbee.readline()#lecture du port série
time.sleep(0.1);
print ("Le port Xbee indique : "+examen_veille)
if(examen_veille=="up"):#si la station envoie up, elle se réveille
can3.delete(ALL)
can2.create_oval(5,5,45,45,fill='green')
can3.create_text(200,24, text ='Vert : station accessible.')
B7.config(text="Desactiver la mise en veille")
forcage_mesures()
Xbee.flushInput()
if(examen_veille=="down"):#si la station envoir down, elle se met en veille
can3.delete(ALL)
can2.create_oval(5,5,45,45,fill='red')
heure_courante = datetime.now()
phrase=' Rouge : station en mode veille à '+str(heure_courante.hour)+' h' +str(heure_courante.minute)+'.
can3.create_text(200,24, text =phrase)
Xbee.flushInput()
if (examen_veille=="T"):
b.stop()
time.sleep(1.0);
heure_courante = datetime.now()
date_heure = str(heure_courante.year) + ";"+str(heure_courante.month) + ";" + str(heure_courante.day)
Xbee.write(date_heure)
print(date_heure)
time.sleep(1.0)
b.start()
Xbee.flushInput()
class MyTimer: #classe définissant un timer
def __init__(self, tempo, target, args= [], kwargs={}):
self._target = target
34
Utiliser une carte Arduino
dans une station météo
self._args = args
self._kwargs = kwargs
self._tempo = tempo
def _run(self):
self._timer = threading.Timer(self._tempo, self._run)
self._timer.start()
self._target()
def start(self):
self._timer = threading.Timer(self._tempo, self._run)
self._timer.start()
def stop(self):
self._timer.cancel()
def lance_mesures() :#lance les mesures
B3.config(state=DISABLED)#le bouton de lancement des mesures est déactivé
B4.config(state=NORMAL)#
print('debut des mesures')
debut=True
#mesures() #une première serie de mesure est prise au début
moment_d_acquisition = datetime.now()
while (moment_d_acquisition.second!=0) :#afin que le démarrage du timer ne prenne effet que lorsque l'heure est de
moment_d_acquisition = datetime.now()
B6.config(state=NORMAL)
a.start()
b.start()
mesures()
def arret_mesures() :
B3.config(state=NORMAL)
B4.config(state=DISABLED)
debut=True
b.stop()
a.stop()
def forcage_mesures():
global forcage
forcage= True
mesures()
def recup_mesures_total():
code_lecture = '3'
lecture_mesures(code_lecture)
def recup_mesures() :
code_lecture = '2'
lecture_mesures(code_lecture)
def lecture_mesures(code_lecture):#permet la récupération des mesures des 48 h précédentes
35
Utiliser une carte Arduino
dans une station météo
b.stop()
time.sleep(1)
Xbee.write(code_lecture)
#Lecture des mesures
time.sleep(2)
M = Xbee.readlines()
#décompose M en autant de lignes qu'il y a d'heures M=M_.split(':')
print M
for n in range(0,len(M)-1):
A = M[n].split(';')
date_=A[2]+"/"+A[1]+"/"+A[0]
A[9]=A[9][0:4]
heure_ = A[3]+"h"+A[4]
mesures_=[date_,heure_,A[6],A[7],A[8],A[9]]
print mesures_
try :
sauvegarde_donnees(mesures_)
except :
tkMessageBox.showwarning('Warning','Accés au fichier impossible ! Veuillez le fermer !')
b.start()
b.start()
def desactive_veille() :#désactive la mise en veille de l'Arduinostation pour la maintenance
global mise_en_veille_active
if (Xbee.isOpen()) :
time.sleep(1)
Xbee.write('7') #code indiquant qu'il faut sortir ou entrer du mode éco
time.sleep(1)
if (mise_en_veille_active==True) :#informe que le mode éco est actif
mise_en_veille_active=False
B7.config(text='Activer la mise en veille')
elif (mise_en_veille_active==False) :#informe que le mode éco est déactivé
mise_en_veille_active=True
B7.config(text='Désactiver la mise en veille')
#######################################################################
#
Programme principal
#
#######################################################################
fenetre_meteo=Tk()#créaction d'une fenêtre graphique
fenetre_meteo.title("Mesures de la station météo")
fenetre_meteo.geometry("800x800")
text1 = StringVar()
text1.set("Activer la recherche du Xbee") #initialise le message : il faut lancer la détection du Xbee
text2 = StringVar()
text2.set("Mesures : ") #
text3 = StringVar()
text3.set("Temps : ")
text_tempe = StringVar()
36
Utiliser une carte Arduino
dans une station météo
text_tempe.set("°C")
text_pression=StringVar()
text_hygro=StringVar()
text_vent=StringVar()
label_tempe =Label(fenetre_meteo,textvariable=text_tempe,relief=RAISED,font=("Sans Serif", "24", "bold"),fg ="red"
label_tempe.place(x=20,y=400) #affichage de la température
label_pression =Label(fenetre_meteo,textvariable=text_pression,relief=RAISED,font=("Sans Serif", "24", "bold"),fg =
label_pression .place(x=20,y=450) #affichage de la pression
label_hygro =Label(fenetre_meteo,textvariable=text_hygro,relief=RAISED,font=("Sans Serif", "24", "bold"),fg ="oran
label_hygro.place(x=20,y=500) #affichage de l'hygrométrie
label_vent = Label(fenetre_meteo,textvariable=text_vent,relief=RAISED,font=("Sans Serif", "24", "bold"),fg ="green"
label_vent.place(x=20,y=550)#affichage de la vitesse du vent
B1=Button(fenetre_meteo,text="Détection du Xbee",command=detection_port_Xbee,state=NORMAL,width=100)#Bo
B1.place(x=20,y=10)#bouton de détection du Xbee
croix=PhotoImage(file='C:/IFE/Météo/documents_contribution/programme python/croix.gif')#image de la croix du bou
B2=Button(fenetre_meteo,image=croix,text='Fermer',command=fermer_programme,state=NORMAL)#Bouton de ferm
B2.place(x=600,y=700)
L1=Label(fenetre_meteo,textvariable=text1)#Label Texte affichant l'état d erecherche du Xbee
L1.place(x = 160, y=50)
B3=Button(fenetre_meteo,text="Lancer les mesures ",command=lance_mesures,state=DISABLED,width=100)#bouton
B3.place(x=20,y=200)#
B4=Button(fenetre_meteo,text="Arrêter les mesures ",command=arret_mesures,state=DISABLED,width=100)#bouton
B4.place(x=20,y=250)
B5=Button(fenetre_meteo,text="SD : récuperer les mesures enregistrées",command=recup_mesures,state=NORMAL,w
B5.place(x=100,y=700)#commande la lecture du fichier du jour en cours
B7=Button(fenetre_meteo,text="SD : récuperer les mesures enregistrées les jours précédents",command=recup_mesure
B7.place(x=100,y=650)#commande la lecture de tous les fichiers
B6=Button(fenetre_meteo,text="Forçage des mesures",command=forcage_mesures,state=DISABLED,width=50)
B6.place(x=100,y=100)#commande manuelle la lecture des mesures
B7=Button(fenetre_meteo,text="Désactiver la mise en veille",command=desactive_veille,state=NORMAL,width=50)
B7.place(x=400,y=100)# Bouton permettant de désactivier la mise en veille du Xbee Distant
L2=Label(fenetre_meteo,textvariable=text2)
L2.place(x = 20, y=300)#Label d'affichage
L3=Label(fenetre_meteo,textvariable=text3)
L3.place(x = 20, y=350)
can1 = Canvas(fenetre_meteo,bg='yellow',height=50,width=700)#canvas d'affichage de l'état de la batterie fond
can1.place(x= 20, y = 600)
can2 = Canvas(fenetre_meteo,height=50,width=50)#canvas d'affichage de l'état de la batterie
can2.place(x=650,y=450)
can3 = Canvas(fenetre_meteo,height=50,width=400)
can3.place(x=400,y=400)
poly2=can1.create_polygon(0, 0, 0, 0,0,50,0,50, fill='red')#canvas d'affichage de l'état de la batterie en rouge si pas asse
txt0=can1.create_text(350, 30, text="charge batterie", font=("Arial"," 16"), fill="blue")#Texte indiquant l'état de la batt
texte1=can1.create_text(0, 12, text="0 V" , font=("Arial"," 12"," italic"), fill="black")
a = MyTimer(60, mesures, ["MyTimer"]) #un timer de 1 h demande la lecture des données et leur enregistrement
b = MyTimer(2, scrutation_mise_en_veille, ["MyTimer"])#un timer de 2 s scrute le port Xbee
fenetre_meteo.mainloop()
37