Arduino et Python

Disons le d'emblée ce billet ne vise qu'à montrer comment Arduino et Python peuvent communiquer. Il s'adresse aux plus téméraires et aux curieux! Vous trouverez sur le Web de multiples ressources à ce sujet. L'idée est simple: vous avez une version de Python disponible sur le réseau et vous désirez piloter l'Arduino. Voici comment faire. [Source: http://www.f-legrand.fr/scidoc/docmml/sciphys/arduino/python/python.html Notons que ce code peut fonctionner à partir d'une clé USB! Seuls le driver Arduino et le logiciel Arduino (quoiqu'il existe un plan B ;)) devront être installés sur le poste.

Le code Arduino

#define PIN_MODE 100
#define DIGITAL_WRITE 101
#define DIGITAL_READ 102
#define ANALOG_WRITE 103
#define ANALOG_READ 104

void commande_pin_mode() {
    char pin,mode;
    while (Serial.available()<2);
    pin = Serial.read(); // pin number
    mode = Serial.read(); // 0 = INPUT, 1 = OUTPUT
    pinMode(pin,mode);
}

void commande_digital_write() {
   char pin,output;
   while (Serial.available()<2);
   pin = Serial.read(); // pin number
   output = Serial.read(); // 0 = LOW, 1 = HIGH
   digitalWrite(pin,output);
}

void commande_digital_read() {
   char pin,input;
   while (Serial.available()<1);
   pin = Serial.read(); // pin number
   input = digitalRead(pin);
   Serial.write(input);
}

void commande_analog_write() {
   char pin,output;
   while (Serial.available()<2);
   pin = Serial.read(); // pin number
   output = Serial.read(); // PWM value between 0 and 255
   analogWrite(pin,output);
}

void commande_analog_read() {
   char pin;
   int value;
   while (Serial.available()<1);
   pin = Serial.read(); // pin number
   value = analogRead(pin);
   Serial.write((value>>8)&0xFF); // 8 bits de poids fort
   Serial.write(value & 0xFF); // 8 bits de poids faible
}

void setup() {
  char c;
  Serial.begin(500000);
  Serial.flush();
  c = 0;
  Serial.write(c);
  c = 255;
  Serial.write(c);
  c = 0;
  Serial.write(c);
}

void loop() {
  char commande;
  if (Serial.available()>0) {
     commande = Serial.read();
     if (commande==PIN_MODE) commande_pin_mode();
     else if (commande==DIGITAL_WRITE) commande_digital_write();
     else if (commande==DIGITAL_READ) commande_digital_read();
     else if (commande==ANALOG_WRITE) commande_analog_write();
     else if (commande==ANALOG_READ) commande_analog_read();
  }
  // autres actions à placer ici
}

Le protocole d'échange est basé sur l'envoi de caractères en 8 bits sur la liaison série. Une séquence d'initialisation 0-255-0 permet d'indiquer que l'arduino est prête.

Il y a un code pour les différentes commandes.

#define PIN_MODE 100
#define DIGITAL_WRITE 101
#define DIGITAL_READ 102
#define ANALOG_WRITE 103
#define ANALOG_READ 104

A chaque commande un certain nombre de paramètres est requis. Pour écrire sur une sortie digitale on enverra trois caractères: 101 (pin) (output). Actionner la sortie 13: 101-13-1

Ce code vous permet d'ajouter des fonctionnalités comme des composants branchés en I2C avec des codes de commandes que vous pourrez ajouter facilement. L'on pourrait également ajouter la lecture d'un capteur température/humidité DHT11/DHT22 (ce sera votre devoir de vacances ;)).

Une classe python est créée pour permettre un interfaçage facile.

commandesPython.py

# -*- coding: utf-8 -*-
import serial
                
class Arduino():
    def__init__(self,port):
        self.ser = serial.Serial(port,baudrate=500000)
        c_recu = self.ser.read(1)
        while ord(c_recu)!=0:
            c_recu = self.ser.read(1)
        c_recu = self.ser.read(1)
        while ord(c_recu)!=255:
            c_recu = self.ser.read(1)
        c_recu = self.ser.read(1)
        while ord(c_recu)!=0:
            c_recu = self.ser.read(1)
        self.PIN_MODE = 100
        self.DIGITAL_WRITE = 101
        self.DIGITAL_READ = 102
        self.ANALOG_WRITE = 103
        self.ANALOG_READ = 104
        self.INPUT = 0
        self.OUTPUT = 1
        self.LOW = 0
        self.HIGH = 1
        
    def close(self):
        self.ser.close()

    def pinMode(self,pin,mode):
        self.ser.write(chr(self.PIN_MODE).encode())
        self.ser.write(chr(pin).encode())
        self.ser.write(chr(mode).encode())

    def digitalWrite(self,pin,output):
        self.ser.write(chr(self.DIGITAL_WRITE).encode())
        self.ser.write(chr(pin).encode())
        self.ser.write(chr(output).encode())

    def digitalRead(self,pin):
        self.ser.write(chr(self.DIGITAL_READ).encode())
        self.ser.write(chr(pin).encode())
        x = self.ser.read(1)
        return ord(x)

    def analogWrite(self,pin,output):
        self.ser.write(chr(self.ANALOG_WRITE).encode())
        self.ser.write(chr(pin).encode())
        self.ser.write(chr(output).encode())

    def analogRead(self,pin):
        self.ser.write(chr(self.ANALOG_READ).encode())
        self.ser.write(chr(pin).encode())
        c1 = ord(self.ser.read(1))
        c2 = ord(self.ser.read(1))
        return c1*0x100+c2

fichier test.py

# -*- coding: utf-8 -*-
import time
from commandesPython import Arduino

port = "COM8"
ard = Arduino(port)
    
ard.pinMode(13,ard.OUTPUT)



for i in range(10):
    print(ard.analogRead(0))
    print(i)
    ard.digitalWrite(13,ard.HIGH)
    time.sleep(0.5)
    ard.digitalWrite(13,ard.LOW)
    time.sleep(0.5)
    
ard.close()

Si la commande en ligne de commande vous semble trop austère, il est possible d'ajouter un contexte fenêtré facilement.

# -*- coding: utf-8 -*-
import time
from commandesPython import Arduino
from tkinter import *

class App(Arduino):
    def __init__(self):
        self.port = "COM8"
        self.ard = Arduino(self.port)
        self.W=200
        self.H=200
        self.root = Tk()
        self.create_interface()
        self.update_clock()
        self.configure()
        self.root.mainloop()

    def send_arduino(self):
        self.ard.digitalWrite(13,self.ard.HIGH)
        time.sleep(0.25)
        self.ard.digitalWrite(13,self.ard.LOW)
        time.sleep(0.25)
      
    def create_interface(self):
        can = Canvas(self.root, width=self.W, height=self.H, bg='ivory')
        can.pack(side=TOP, padx= 5, pady= 5)
        btn1 = Button(self.root, text="Arduino", command = self.send_arduino)
        btn1.pack(side=LEFT)
        self.text1=Label(self.root, text="A0: ", fg="red")   
        self.text1.pack()

    def configure(self):
        self.ard.pinMode(13,self.ard.OUTPUT)
		
    def update_clock(self):
        now = time.strftime("%H:%M:%S")
        self.text1.configure(text= self.ard.analogRead(0))
        self.root.after(1000, self.update_clock)

app=App()

Deuxième exemple, plus évolué: un programme python qui embarque un serveur twisted pour contrôler l'Arduino. Exemple de commandes:

<A href=“http://localhost:8080/set/digital/5/on”>Allumer la sortie digitale 5</A>
<A href=“http://localhost:8080/set/digital/5/off”>Eteindre la sortie digitale 5</A>
<A href=“http://localhost:8080/get/analogic/0”>Lire l\'entrée analogique A0</A>
<A href=“http://localhost:8080/temp/dht/4/11”>Lire la température sur le dht11<A>

"""
Programme de gestion d'une Arduino à partir de Python
A partir de tkinter, twisted et commandesPython (R)
MrT - sebastien.tack@ac-caen.fr
"""
from tkinter import *
from twisted.internet import tksupport, reactor
from twisted.web import server, resource
from twisted.internet import reactor, endpoints
from commandesPython import Arduino
import serial.tools.list_ports
import sys

class Counter(resource.Resource):
    isLeaf = True
    numberRequests = 0
    
  
    def __init__(self, root):
        """
        Constructeur de la classe
        """
        self.root = root
        self.port = None
        ports = list(serial.tools.list_ports.comports())
        for p in ports:
            if "Arduino" in p[1]:
                self.port =str(p[1][p[1].find("(")+1:p[1].find(")")])  
        try:    
            self.ard = Arduino(self.port)
                      
        except:
            print("Branchez l'arduino, Svp!")
            self.root.destroy()
            sys.exit(1)
        self.make_interface()
 
    def make_interface(self):
        """
        Création de l'interface tk
        """
        self.text2 = Label(self.root, text="Voir http://localhost:8080/")
        self.text2.pack()    
    
   
    def send_arduino(self,port,value):
        """
        Envoi des ordres en sortie sur Arduino
        """
        self.ard.digitalWrite(port,value)
    
    def render_GET(self, request):
        """
        Gestion des réponses GET
        """
        #self.text2.configure(text= "ok")
        self.numberRequests += 1
        #trouver path
        content= ""
        req = request.uri
        if b'favicon' in req:
            pass
            
        if req==b'/':
            
            content ='<!DOCTYPE html><html><head><meta charset="utf-8" /></head><body>'        
            content  += "Exemples ..."
            content += "<br />"
            content += u'<br /><A href="http://localhost:8080/set/digital/5/on">Allumer la sortie digitale 5</A>'
            content += u'<br /><A href="http://localhost:8080/set/digital/5/off">Eteindre la sortie digitale 5</A>'
            content += u'<br /><A href="http://localhost:8080/get/analogic/0">Lire l\'entrée analogique A0</A>'
            content += u'<br /><A href="http://localhost:8080/temp/dht/4/11">Lire la température sur le dht11 en pin 4</A>'
            content += "</body></html>"

        if b'/set/digital' in req:
            
            ordres = req.split(b'/')
            pin = int(ordres[3])
            mode = (ordres[4] == b'on')
            self.ard.pinMode(pin,self.ard.OUTPUT)
            self.send_arduino(pin, mode)
            content ='<!DOCTYPE html><html><head><meta charset="utf-8" /></head><body>'
            content = "Sending "+str(mode)+" to pin "+str(pin)
            content += "</body></html>"

        if b'/get/analogic' in req:
            
            ordres = req.split(b'/')
            pin = int(ordres[3])
            value = self.ard.analogRead(pin)
            content ='<!DOCTYPE html><html><head><meta charset="utf-8" /></head><body>'
            content += "Pin "+str(pin)+" is to "+str(value)
            content += "</body></html>"
            
        if b'/temp/dht' in req:
            
            ordres = req.split(b'/')
            pin = int(ordres[3]) 
            mode = int(ordres[4])            
            self.ard.pinMode(pin,self.ard.INPUT)
            content ='<!DOCTYPE html><html><head><meta charset="utf-8" /></head><body>'
            content += u"Température "+self.ard.dht_read(pin,mode)+" °C"
            content += "</body></html>"             
                    
        request.setHeader(b"content-type", b"text/html")
        return content.encode("utf-8")

"""
Début du programme
"""        
        
try:     
    site = Counter(Tk())
    tksupport.install(site.root)
    endpoints.serverFromString(reactor, "tcp:8080").listen(server.Site(site))
    reactor.run()
except:
    pass

On peut maintenant construire des pages HTML et réaliser des requêtes Ajax en jQuery sur ces URL pour piloter une Arduino à partir de pages Web. ;)