LoginSignup
5
4

More than 1 year has passed since last update.

LabVIEWとPythonの連携

Posted at

はじめに

LabVIEWのサブスクリプト移行により、計測部分はこれまで使ってきたLabVIEWで、解析部分だけをPythonで行えないか?との相談を受けた。(そうするとLabVIEWのサブスク費用を抑えることができる?解析モジュールなどのアドオンを使わないですむ。)相談では、リアルタイム処理(数msオーダーの処理)は不要とのことでした。そこで2つの解決策を考えました。

  • LabVIEWからPythonを呼び出して処理
    LabVIEWにある(1)Pythonノード、(2)コマンドプロンプト、(3)HTTPクライアントなどを利用してPythonとの連携が考えられます。(1)のPythonノードは使ったことがないのでわかりませんが、LabVIEWでPythonコードを活用する の解説を見ると制約が多い印象を受けます。(2)コマンドプロンプトの場合、Python側では sys.argvを使って引数を渡すようにプログラムする必要があります。(3)HTTPの場合、Python側でFLASKやFastAPIなどでWebアプリを作る必要があります。
    これらの方法を利用するには、そこそこのLabVIEWとPythonの両方の知識が必要になります。
    コマンドプロンプトを使った参考記事はこちら(2017年の記事)。
    LabVIEWとPythonでQRコードを作る

  • ファイルで渡す
    古典的な方法ですが、ファイルでの受け渡しが一番簡単です。この場合、ファイルの受け渡しルールだけを決めれば、それぞれの言語に精通している人で別々に作成できます。
    例えば、
    LabVIEW側であるフォルダーに計測データをセーブ(例えば、ヘッダー付きCSV)して、Pythonで処理したデータファイルを別のフォルダーに保存(こちらもヘッダー付きCSV)して、LabVIEW側でそのファイルを読み込む方法が考えられます。ある決められたフォルダーを監視して、ファイルがあれば処理して、他のフォルダーに処理した結果とTargetにあるファイルを移動します(解析デーモン)。

今回の要件から考えるとファイルで渡す方式が適切だと考えられましたので、このプログラム例を参考までに示します。

解析監視GUI (解析デーモン)

Target Folder(処理するファイルが入るフォルダー)とSave Folder(処理したファイルの保存と、Targetに入っていたファイルを移すフォルダー)を用意します。
Target Folderを巡回監視して、ファイルがあれば処理して、Save Folderに処理した結果とTargetにあるファイルを移動します。PysimpleguiでGUIをつけています。(Pythonが入っていないPCにはPyInstallerなどでExe化して配布します。)

q03.PNG

プログラム環境

Windows 10 Pro 64 bit
Python3.7以上
Anacondaを利用

解析監視GUIの説明

Target Folder: 処理するファイルが入るフォルダー
Save Folder:Targetに入っていたファイルを移すフォルダーとグラフ画像
File Type:装置によってFile Typeが異なるので、Type1 とType2のどちらかを選ぶ
Loop start:TargetFolderの監視開始
Loop Stop:監視停止
Cancel:プログラム終了

動作説明

TARGET Folderに data1.csvファイルが入っています。Loop startを押すとそのファイルが処理されて、Save Folderに移されます。
q04.PNG

Cancelを押すとプログラムは終了して結果をまとめたCSVファイルが作成されます。

なおLoop stopで監視処理を止めます。Loop startで再開できます。
q05.PNG

プログラムと説明

"""
Targetフォルダーに解析対象ファイルがある場合、解析を行う
解析対象ファイルをSaveフォルダーに移動する
解析結果の画像、解析のまとめCSVファイルを作成してSaveフォルダーに格納

解析対象ファイルのフォーマットが異なる(例えば、xyの並びが異なる、ヘッダーがないなど)場合でも選べる様にする

初期設定ファイル(Toml形式)でTarget,Saveフォルダーをあらかじめ指定する。

"""

from pathlib import Path
import shutil
import time
import toml
import pandas as pd
import numpy as np
import PySimpleGUI as sg


# --- ファイルを解析するクラス (別ファイルに作成してもよい)
import datetime

def now_datetime(type=1):
    """
    日時を文字列で返す
    type1:通常表示 "%Y-%m-%d %H:%M:%S"
    type2:"%Y%m%d%H%M%S"
    type3:"%Y%m%d_%H%M%S"
    type4:Base_file_nameで使用する形式 "%Y%m%d%H%M"
    elae:日付のみ "%Y%m%d"
    :return:
    """
    now = datetime.datetime.now()
    if type == 1:
        now_string = now.strftime("%Y-%m-%d %H:%M:%S")
    elif type == 2:
        now_string = now.strftime("%Y%m%d%H%M%S")
    elif type == 3:
        now_string = now.strftime("%Y%m%d_%H%M%S")
    elif type == 4:
        now_string = now.strftime("%Y%m%d%H%M")
    elif type == 5:
        now_string = now.strftime("%m%d_%H:%M:%S")
    elif type == 6:
        now_string = now.strftime("%Y%m%d")    
    else:
        now_string = now

    return  now_string

class Analysis():
    """
    example:
        fname = r'./TARGET/data1.csv'
        test = Analysis(fname)
        test.read_file()
        test.estimate()
        print(test.meta)

    """
    
    def __init__(self,file):
        self.file = Path(file)
        
    def read_file(self):
        
        #Pandasのcsv読み込みを例に
        self.df = pd.read_csv(self.file)
        
    def estimate(self):
        # 解析結果を辞書型に格納 
        total =  np.sum(self.df.iloc[:,1]*2) # Intensityを2倍にして和を計算
        
        self.meta = {'file_name':self.file.stem,'data':total}     

# Analysis Classを継承しています。異なる動作をさせたいメソッドのみ記述
class AnalysisB(Analysis):
    
    def read_files(self):
        pass
        # with open(self.file) as f:
        #     s = f.read()
        # self.data = s 
 
       
# ------ ここからGUIプログラム

sg.theme('Light Blue 2')

# 設定ファイルをTOMLで用意している場合
try: 
    with open('./setfile.toml') as f:
        set_obj = toml.load(f)
except: #tomlファイルがない場合
    set_obj = {"target_folder":'',"save_folder":'','file_type':'type1'}
    

layout = [[sg.Text('Auto Analysis')],
            [sg.Text("Target Folder"), sg.InputText(f'{set_obj["target_folder"]}',size=(80,4), key="-a_folder-")],
            [sg.Text("Save   Folder"), sg.InputText(f'{set_obj["save_folder"]}',size=(80,4), key="-s_folder-")],
            [sg.Text('File type'), sg.Radio('Type A', group_id='0',  default=set_obj['file_type'] == 'type1', key='-1-'), 
             sg.Radio('Type B', group_id='0',default=set_obj['file_type'] == 'type2',key='-2-')],
            [sg.Button('Loop start',key='-begin-'), sg.Button('Loop stop',key='-stop-')],
            [sg.Cancel()],
            [sg.Output(size=(80, 20))],
          ]

window = sg.Window('Auto Analysis', layout, location=(100, 100), size=(500, 500), resizable=True) 

read_flag = 0

# 解析した値を保存するため。(適宜変更)
date_res = []
file_res = []
total_res = []

while True:
    event, values = window.read(timeout=1000,timeout_key='-timeout-')
    # 1000msおきに-timeout- event が呼び出されます(timeoutを適宜変更)

    if event == '-begin-':
        print('Loop Start')
        read_flag = 1
        read_path = Path(values['-a_folder-'])
        
        # Saveフォルダーがない場合は作成される。
        save_path = Path(values['-s_folder-'])
        save_path.mkdir(exist_ok=True)

    elif event == '-stop-':
        print('Loop Stop')
        read_flag = 0
    
    elif event in '-timeout-':

        if read_flag == 1:
            files = list(read_path.glob('*'))
            # print(files)
            
            if len(files) != 0:
                # 必要かどうか検討(対象フォルダーに完全にファイルが移動できていない場合には、エラーになる可能性があるので0.5秒待っている)
                time.sleep(0.5) 
                    
                for fi in files:
                    if values['-1-']: # Type A
                        tmp_ins = Analysis(fi)
                    elif values['-2-']: # Type B 
                        tmp_ins = AnalysisB(fi) 
                    
                        
                    tmp_ins.read_file() 
                    tmp_ins.estimate()
                    
                    f_name = tmp_ins.meta["file_name"]
             
                    
                    # Targetフォルダーにあるファイルを移動する
                    new_p = save_path / fi.name
                    
                    # 移動するフォルダーに同じ名前のデータがあるとエラーになるため、ファイルがあるか確かめる
                    # ファイルがある場合は削除
                    if new_p.exists(): 
                        new_p.unlink()
                        
                    # shutil.move(fi,save_path) # Python 3.7だとエラーになる。
                    # https://tec.citrussin.com/entry/2019/03/24/201649
                    shutil.move(fi,new_p) 
                    
                    # fi.rename(new_p)
                    treat_date = now_datetime(type=3)
                    print(f'{treat_date}, Calculate, fname : {f_name}')
                    
                    date_res.append(treat_date)
                    file_res.append(fi.name)
                    total_res.append(tmp_ins.meta["data"])

    
    elif event in (None, 'Cancel'):
        # 処理した結果をまとめてCSVに出力する。
        res_dict = {'date': date_res,'file':file_res, 'data':total_res}
        df = pd.DataFrame(res_dict)
        csv_save_path = save_path / f'{now_datetime(type=4)}.csv'
        df.to_csv(str(csv_save_path), index=False)

        break
    
window.close()

初期設定ファイル

setfile.toml

# setfile.toml
target_folder = './TARGET'
save_folder = './SAVE'

# default read file format
# input 'type1' or 'type2'

file_type = 'type1'

なお、これらのプログラムとデータは、
deamon_gui
にあります。

以上、PythonとLabVIEWの連携の参考になれば。

5
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
4