# MAPO-IOB-WIN-PY Progetto per porting script Python della comunicazione con schede IOB seriali (versione iniziale 2006). Funzionato fino al 2024 tramite PC winXP locali e seriale locale, poi passati a Moxa + seriali virtuali e IOB dedicato (dove chiama Moxa sia x part program e d_cx che x IOB-WIN python) ## Specifiche tecniche il progetto prevede il sorgente python che viene poi compilato, tramite PyInstaller, in un EXE windows che è monolitico e si occupa di tutto il programma (semplificando la gestione) E' un progetto separato e a se stante rispetto ai normali IOB-WIN-* perché - non è compilato da VisualStudio - non usa dotNet Framework (ma appunto python) - non comunica via rete direttamente ma tramite seriale (remotata via Moxa dal 2024) ## Setup preliminare Va selezionato in VSCode l'interprete python corretto (3.11 al momento della stesura), tramite **CTRL+Shift+P** e poi *Python: Select Interpreter* Vanno installate eventuali dipendenze (es **pip install redis**) x moduli non standard ## Modifiche principali Le modifiche principali per passare dalla versione WinXp (interpretata) alla nuova (compilata) hanno riguardato * migrazione Python2 (2.7) --> Python3 (3.11) * impiego di VSCode come editor e debugger soluzione * revisione gestione parser e lettura configurazione * nuova configurazione logfile (con rotate gestito dalla libreria) * introduzione librerie x Redis * gestione oggetto stato IOB da serializzare su Redis x comunicazione WatchDog con IOB-WIN * intruzione compilazione del python con PyInstaller * configurazione compilazione + copia conf tramite script bat (**createExe.bat**) ## Creazione exe Impieghiamo PyInstaller. sito: https://pyinstaller.org/en/stable/usage.html Deve essere aggiunto tramite pip sulla macchina di build ```python pip install PyInstaller ``` Viene generato exe con il comando PyInstaller che nella versione minimale è qualcosa come ```python python -m PyInstaller IOB-WIN-PSER-TEST.py ``` Se si volesse generare un unico file exe + icona la sintassi diventa ```python python -m PyInstaller --onefile --icon SteamWare.ico IOB-WIN-PSER.py ``` Recupero ed estrazione file versione ```python pyi-grab_version.exe .\dist\IOB-WIN-PSER-vers.exe ``` Reimpostazione file versione ```python pyi-set_version.exe file_version_info_new.txt dist\IOB-WIN-PSER.exe ``` Il file exe + le varie librerie accessorie saranno salvate in **dist/nome_progetto/** NB: bisogna poi a mano copiare i file di conf, ma non nella cartella principale ma in **_internal**, ovvero **dist/nome_progetto/_internal** Esempio bat finale: ```bat REM script creazione EXE da python a mano REM setup preliminare virtual environment x lavorare sul progetto se mancasse REM python -m venv venv REM attivazione virtual env specifico REM .\venv\Scripts\Activate.ps1 REM install componente REM pip install PyInstaller REM estrazione file version (come file_version_info.txt) REM pyi-grab_version.exe .\dist\IOB-WIN-PSER-vers.exe @REM python -m PyInstaller --onefile --icon SteamWare.ico IOB-WIN-PSER-TEST.py python -m PyInstaller --onefile --icon SteamWare.ico IOB-WIN-PSER.py REM effettua copia delle conf x poter avviare... xcopy CONF\*.cfg dist\CONF\ REM editing file versione con valori vers corrente x dataOra REM reinserimento info versione da file editato (file_version_info_new.txt) REM pyi-set_version.exe file_version_info_new.txt dist\IOB-WIN-PSER.exe ``` ## Integrazione in yaml di CI/CD A partire da gennaio 2025 è stato separato il progetto e costruito un apposito pipeline di compilazione dello stesso. Per farlo nel file yaml sono specificati i comandi per - impiegare il corretto virtual environment - compilare l'eseguibile dallo script python - sistemare il numero di versione nel file template - impostare il manifest della versione windows nell'exe finale - zippare ed inviare a nexus versione compilata - registrare sul liman disponibilità della release compilata ## Lancio dell'eseguibile L'eseguibile è richiamabile direttamente o tramite BAT, si aspetta in ingresso un parametro, ovvero il nome del file *.cfg con cui avviarsi, che conterrà * configurazioni base x gestione letture varie * parametri connessioni porta comm(virtuale tramite MOXA) * parametri accessori x log, ... ## Integrazione con IOB-MAN Per poter venire integrata con IOB-MAN è necessario che ci sia una specifica modalità di comunicazione, tramite Redis, che ritorna lo stato dell'adapter al manager per effettuare le procedure di watchdog ed evitare riavvio automatico per mancata comunicazione. Si è impiegata questa calsse helper serializzata e salvata su redis con un thread separato ```python class IobWinStatus: def toJSON(self): return json.dumps(self, default=lambda o: o.__dict__, sort_keys=True, indent=4) #stream = 'cse' # Class Variable def __init__(self,codIob,iobType): self.CodIob = codIob self.IobType = iobType self.counterIOB = 0 self.counterMAC = 0 self.freeNotes = '' self.lastDataIn = '' self.lastDataOut = '' self.lastUpdate = '' self.online = False self.queueAlLen = 0 self.queueEvLen = 0 self.queueFlLen = 0 self.queueMsLen = 0 self.queueRawTransfLen = 0 self.queueUlLen = 0 self.semIn = 'ND' self.semOut = 'ND' ``` Periodicamente si aggiornano indicazioni dei valori **lastDataIn** (Out,...) e il boolean di **online** e poi serializza su redis ## Configurazione IOB-MAN ### Vers <= 3.6 IOB Man fino alla versione 3.6 va configurato in modo che vada correttamente a lanciare gli eseguibili **IOB-WIN-PSER** (ovvero del compilato Python-Seriale), andando ad agire sul file *.config: ```XML ... ... ... ... ``` ottenendo alla fine qualcosa come ```XML ``` ### Vers >= 3.7 Dalla versione 3.7 è permesso il lancio di vari applicativi differenti da un unico IOB-MAN. Per farlo correttamente va comunque imposta il (nuovo) file di configurazione json con le apposite sezioni di chiamata, ad esempio ```json { "ListIOB": { "L003": "IOB-WIN-PSER", "03": "d_cx", } "ListTarget": { "IOB-WIN-PSER": { "ExeName": "IOB-WIN-PSER", "ExePath": "C:\\Steamware\\IOB-WIN-PSER\\IOB-WIN-PSER.exe", "LogDir": "C:\\Steamware\\IOB-WIN-PSER\\logs\\", "NLogPath": "C:\\Steamware\\IOB-WIN-PSER\\CONF\\NLog.config", "BaseArgs": "" }, "d_cx": { "ExeName": "d_cx", "ExePath": "C:\\Steamware\\d_cx\\d_cx.exe", "LogDir": "C:\\Steamware\\d_cx\\logs\\", "NLogPath": "C:\\Steamware\\d_cx\\NLog.config", "BaseArgs": "" } } } ``` Dove nella prima aprte è indicato ogni IOB e quale exe chiamare, mentre nella seconda parte sono specificati i parametri di avvio dell'exe e in particolare i **BaseArgs** con cui avviare (per IOB-WIN-PSER sono vuoti come per d_cx a differenza dei normali IOB-WIN-*) ## Approfondimenti ### Link vari impiegati per la preparazione Di seguito pagine link varie usate x sistemare progetto * https://pyinstaller.org/ * https://pyinstaller.org/en/stable/usage.html * https://datatofish.com/executable-pyinstaller/ Valutati ma non impiegati * https://github.com/Nuitka/Nuitka * https://pypi.org/project/py2exe/ ## Version Date | Vers | Note ---------|----------|--------- 2006.*.* | 1.0.*.* | Prima versione in DotNet, PC locale WinXP con stati locali 2018.*.* | 2.0.*.* | Versione Python locale vers 2.7, sempre WinXP 2024.06.01 | 2.8.2.* | Versione Python vers 3, Compilata con PyInstaller, per gestione tramite IOB-MAN + Moxa 2025.01.27 | 2.9.*.* | Separazione progetto a se stante, versione con gestione del numero versione x File e Applicazione (x integrazione con LiMan e EgwAppControlCenter)