(Italiano) Raspberry Pi – Un esempio di applicazione della TS-CNS
Per il ponte del due giugno ho racimolato qualche “pezzo di ferro” e realizzato con il Raspberry Pi un piccolo progetto che prevede l’uso di un lettore di Smart Card e di altri componenti.
L’idea alla base della “ricetta” è quella di realizzare un sistema di sicurezza tale per cui è possibile attivare una serie di relè solo dopo che l’utilizzatore del sistema esegua con successo l’autenticazione tramite la propria TS-CNS (Tessera Sanitaria – Carta Nazionale dei Servizi). Sul sito dell’Agenzia per l’Italia digitale (AgID) sono pubblicate tutte le informazioni di dettaglio sulla Carta Nazionale dei Servizi.
Nel corso di questo articolo vi mostrerò come ho realizzato il progetto, sia dal punto di vista hardware sia dal punto di vista software. Se siete impazienti di vedere il risultato ottenuto, vi invito a vedere il video Un sistema di accesso basato sulla Smart Card TS-CNS e Raspberry Pi che ho pubblicato sul mio canale YouTube.
Questo progetto nasce dalla composizione dei seguenti “pezzi di ferro”.
- Raspberry Pi 3 Model B+
- Display LCD 16×2
- Modulo quattro relè
- KeyPad 16 caratteri (4×4)
- Lettore Smart Card Bit4id
Recentemente ho utilizzato il display LCD e il modulo da quattro relè nel progetto Un primo maggio 2020 a base di Raspberry Pi, Bot Telegram, Display LCD e Relè che vi invito a leggere. Quelli che vedete nella figura a seguire sono i componenti utilizzati per la realizzazione del progetto (a meno della tastiera).
1. Schema elettrico della soluzione
Dando per scontato che abbiate tutto il materiale hardware a vostra disposizione (vedi capitolo 2. Requisiti), possiamo disporre tutti collegamenti tra i vari componenti hardware che compongono la soluzione, così come indicato dallo schema elettrico mostrato a seguire.
Al fine di facilitare i collegamenti ho indicato nelle note quali sono i pin (BCM) che ogni componente utilizzerà. Per quanto riguarda il layout del connettore J8 del Raspberry Pi, fate riferimento alla documentazione ufficiale su GPIO (general-purpose input/output).
2. Requisiti
Per raggiungere l’obiettivo abbiamo bisogno di raccogliere un pochino di materiale, dobbiamo fare una sorta di lista della spesa, sia in termini hardware, sia in termini software. Partiamo dal basso, con la lista dell’hardware.
- Raspberry Pi: ho utilizzato la versione 3 Model B+ che sarà il riferimento di questo articolo. Per coloro che hanno a disposizione il Raspberry Pi 2 o il Raspberry Pi 4, non dovrebbe essere un problema.
- Modulo Relè: ho utilizzato il modello Elegoo 4 Channel DC 5V Modulo Relay con accoppiatore ottico. A quanto pare è difficile trovare questo modello, al suo posto, anche su Amazon, è stato sostituito con il modulo da otto. In ogni caso per questo esperimento va benissimo anche il modulo con due o al peggio un solo Relè.
- Display LCD: ho utilizzato un display LCD I2C 1602 16X2.
- KeyPad 16 caratteri (4×4): ho scelto questo tastierino con lettere e caratteri “speciali” per il motivo che vedremo dopo.
- Breadboard, GPIO Extender, Cavi Maschio a Femmina Dupont e Cavi Maschio a Maschio Dupont utilizzati per realizzare i collegamenti tra il Raspberry Pi, il modulo LCD, il modulo Relè e il KeyPad.
Tutti i componenti hardware possono essere acquisitati separatamente, però, se l’argomento vi appassiona, consiglio l’acquisito dello Starter Kit Freenove RFID per Raspberry Pi. All’interno del kit sono inclusi gli ultimi elementi della nostra lista, oltre ad altri numerosi componenti.
Per quanto riguarda il software abbiamo bisogno di:
- Raspbian OS: a meno di applicazioni particolari, questo sistema operativo basato su Debian è ottimizzato per la piattaforma Raspberry Pi. L’ultima versione disponibile è Raspbian Buster. Sul mio Raspberry Pi 3 B+ ho installato quest’ultima release. Per maggiori informazioni sull’installazione del sistema operativo, fare riferimento alla guida ufficiale (Installing operating system images). Questo sarà comunque il sistema operativo di riferimento.
- Python 3.x: sul mio Raspberry Pi ho installato sia la versione 2.7 (quella di default) sia la versione 3.7, quest’ultima disponibile OOTB con l’ultima versione di Raspbian OS.
- pad4pi: modulo Python per la gestione del KeyPad.
- Moduli software display LCD: moduli Python per utilizzare il display LCD. Utilizzeremo i moduli scritti da Freenove e reperibili dal loro repository GitHub.
- PC/SC Middleware: Middleware per la gestione e comunicazione per le Smart Card e Smart Card Reader.
- OpenSC: OpenSC fornisce una serie di librerie e utilità per lavorare con le Smart Card. Il suo focus principale è sulle schede che supportano le operazioni crittografiche e ne facilitano l’uso in applicazioni di sicurezza come l’autenticazione, la crittografia della posta e le firme digitali. OpenSC implementa le API standard su Smart Card, ad es. API PKCS#11, Minidriver per Smart Card di Windows e macOS CryptoTokenKit.
Possiamo verificare che il software attualmente installato sia conforme ai requisiti accedendo alla console del Raspberry Pi ed eseguendo i comandi a seguire.
# Verifica della Release del Sistema Operativo $ cat /etc/os-release # Verifica della Release del Kernel $ uname -a
A seguire è mostrato l’output dei comandi precedenti eseguiti sul mio Raspberry Pi.
# Output del comando cat /etc/os-release PRETTY_NAME="Raspbian GNU/Linux 10 (buster)" NAME="Raspbian GNU/Linux" VERSION_ID="10" VERSION="10 (buster)" VERSION_CODENAME=buster ID=raspbian ID_LIKE=debian HOME_URL="http://www.raspbian.org/" SUPPORT_URL="http://www.raspbian.org/RaspbianForums" BUG_REPORT_URL="http://www.raspbian.org/RaspbianBugs" # Output del comando uname -a Linux amusarra-rpi 4.19.97-v7+ #1294 SMP Thu Jan 30 13:15:58 GMT 2020 armv7l GNU/Linux
Per quanto concerne la versione di Python, a meno di casi particolari, su Raspbian OS dovrebbero essere installate due release di Python, la 2.x e la 3.x. La specifica minor version dipende dalla versione del sistema operativo. Nel caso di Raspbian OS Buster le versioni esatte sono:
- Python 2.7.16 (attiva per default)
- Python 3.7.3
Sul mio Raspberry Pi ho reso di default la versione 3.7.3 di Python. Per rendere l’ultima versione di Python di default, è più che sufficiente aggiungere i seguenti alias su file ~/.bashrc
alias python='/usr/bin/python3' alias pip=pip3
Una volta concluse le verifiche che riguardano Python, possiamo passare all’installazione del resto dei componenti software.
$ sudo pip install pad4pi $ sudo apt-get install pcscd $ sudo apt-get install libccid $ sudo apt-get install opensc
A questo punto, che i requisiti hardware e software di base sono soddisfatti, vediamo com’è strutturato il progetto software.
3. Organizzazione del progetto
Il progetto software è stato realizzato in Python e tutto il codice sorgente è disponibile all’interno del repository GitHub amusarra/raspberry-pi-access-via-ts-cns. La figura a seguire mostra il contenuto del repository di progetto.
Il progetto è così organizzato:
- modules: questa directory contiene i moduli Python per l’utilizzo del display LCD 16×2
- PCF8574.py: modulo per la gestione del bus i2c
- Adafruit_LCD1602.py: funzioni ad alto livello per le operazioni sul display LCD
- scripts: questa directory contiene lo script Python parse-gov-certs.py il cui scopo è il download dei certificati Governativi Italiani, e lo script bash auto-update-gov-certificates.sh il cui scopo è aggiungere sul sistema i certificati Governativi Italiani. Aggiungere questi certificati è fondamentale affinché il processo di validazione del certificato contenuto all’interno della TS-CNS trovi i certificati validi della catena di certificazione (o certificate path);
- test_keypad_pin_lcd.py: script Python per il test del funzionamento del solo KeyPad;
- verify_ts_cns_pin.py: script Python per la verifica del codice PIN della TS-CNS;
- activate_relay_via_pin_code.py: script Python che permette l’attivazione dei relè inserendo il codice PIN (1234), senza quindi interazione con la TS-CNS;
- activate_relay_via_ts_cns_pin.py: script Python che permette l’attivazione dei relè inserendo il codice PIN della TS-CNS.
Gli script verify_ts_cns_pin.py e activate_relay_via_ts_cns_pin.py sono quelli che interagiscono con il lettore di Smart Card e la TS-CNS. Il resto degli script sono stati realizzati con il solo scopo di verificare la corretta funzionalità del KeyPad e del modulo Relè, e accertarsi quindi che i collegamenti tra i vari componenti stiano funzionando correttamente.
Cerchiamo adesso di capire insieme quali sono i punti importarti dello script activate_relay_via_ts_cns_pin.py che permettono la verifica del codice PIN e del certificato digitale della TS-CNS.
Quella a seguire è la definizione della funzione check_pin(key). Questa funzione è responsabile della verifica del codice PIN immesso tramite il KeyPad. La verifica del codice PIN è eseguita utilizzando direttamente il tool pkcs11-tool. Ricordo che questo tool fa parte del pacchetto d’installazione di OpenSC (che abbiamo installato in precedenza).
# Check entered PIN code def check_pin(key): global entered_pin command_to_check_pin_cns = "pkcs11-tool --login --test --verbose --pin " + entered_pin if len(entered_pin) >= 8 or key == "#": lcd.clear() lcd.message("Check PIN CNS...\n") p = subprocess.Popen(command_to_check_pin_cns, shell=True, stdout=subprocess.PIPE) (output, err) = p.communicate() p_status = p.wait() print("Check PIN: ", output) if p_status == 0: correct_pin_entered() if validate_client_certificate(): select_relay_to_activate() else: incorrect_pin_entered()
Quella a seguire è invece la definizione della funzione validate_client_certificate, responsabile della validazione del certificato digitale contenuto all’interno della TS-CNS. Ricordo che il certificato digitale è rilasciato specificatamente al proprietario della TS-CNS. La verifica del certificato è realizzata attraverso l’uso del tool pkcs15-tool (che fa parte di OpenSC) e del tool OpenSSL.
# Validate the client certificate def validate_client_certificate(): command_to_validate_certificate = "pkcs15-tool -v -r 01|openssl verify" lcd.clear() lcd.message("Check CNS Cert..\n") p = subprocess.Popen(command_to_validate_certificate, shell=True, stdout=subprocess.PIPE) (output, err) = p.communicate() p_status = p.wait() print("Check Client Certificate: ", output) if p_status == 0: print("TS-CNS Client Certificate validation passed") lcd.message("Passed") return True else: print("TS-CNS Client Certificate validation failed") lcd.message("Failed") return False
Per la semplicità del progetto ho preferito utilizzare direttamente i tool messi a disposizione da OpenSC. In contesti più complessi, dove magari è richiesto un maggior controllo sul lettore e sulla Smart Card, è possibile pensare (restando in Python) all’uso di pyscard – Python for Smart Cards.
4. Quick Start
Supponendo che abbiate montato tutto secondo lo schema elettrico indicato e che il vostro Raspberry Pi sia connesso alla rete internet, possiamo procedere con i seguenti step.
- Accesso al Raspberry Pi via SSH o direttamente dal console. Consiglio di accedere e operare con utenza diversa da root.
- Clonazione del repository amusarra/raspberry-pi-access-via-ts-cns.
- Aggiornamento dei certificati di sistema.
- Esecuzione del test di accesso e attivazione relè.
$ cd -- $ git clone https://github.com/amusarra/raspberry-pi-access-via-ts-cns.git $ cd raspberry-pi-access-via-ts-cns $ cd scripts $ sudo ./auto-update-gov-certificate.sh
La figura a seguire mostra l’output che dovreste ottenere eseguendo lo script auto-update-gov-certificates.sh che aggiunge i certificati Governativi Italiani al sistema.
L’aggiunta dei certificati Governativi Italiani al sistema è importante ai fini della validazione del certificato digitale presente sulla TS-CNS. Nel caso questa operazione non venisse eseguita, il processo di autenticazione non andrà mai a buon fine, anche nel caso in cui il PIN inserito fosse corretto.
A questo punto è tutto pronto per iniziare i test di funzionamento della soluzione. Consiglio di procedere con l’esecuzione degli script Python, con lo stesso ordine mostrato nel capitolo 3. Organizzazione del progetto. Per eseguire gli script sarà sufficiente utilizzare la sintassi ./nome_script_python.py
.
La figura a seguire mostra l’esecuzione e l’output dello script verify_ts_cns_pin.py che in questo caso evidenzia l’errato inserimento del codice PIN e di conseguenza l’accesso per l’attivazione dei relè non è consentito.
Le figure a seguire mostrano invece le varie fasi del processo di autenticazione: inserimento del PIN, validazione PIN e certificato digitale, accesso per l’attivazione dei quattro relè tramite il KeyPad.
5. Conclusioni
Questo è l’articolo che va a completare il video tutorial Un sistema di accesso basato sulla Smart Card TS-CNS e Raspberry Pi pubblicato sul mio canale YouTube verso la fine del mese di giugno.
In questo articolo abbiamo avuto modo di vedere quello che si può realizzare con un Raspberry Pi, un lettore di Smart Card, un KeyPad 4×4, un modulo da quattro relè, un display LCD 16×2 e la propria TS-CNS (Tessera Sanitaria – Carta Nazionale dei Servizi). Nel caso vorreste ancora di più, vi invito a leggere l’articolo Raspberry Pi e Smart Card Mifare Classic 1K: Realizzare un sistema di accesso.
Spero di aver acceso la tua fantasia per realizzare idee simili a quella che abbiamo appena vista insieme. Se ti fa piacere condividi pure attraverso i commenti, le possibili idee che questo articolo hanno scatenato.