Merge
authorEdouard Tisserant <edouard.tisserant@gmail.com>
Wed, 23 May 2018 20:22:45 +0200
changeset 2017 c76d4be6f438
parent 2011 64268e87613e (diff)
parent 2016 2a8cd24a14ca (current diff)
child 2021 bcf346f558bd
child 2166 5ce6d08ff2c7
Merge
--- a/ConfigTreeNode.py	Wed May 23 12:13:48 2018 +0300
+++ b/ConfigTreeNode.py	Wed May 23 20:22:45 2018 +0200
@@ -274,15 +274,14 @@
             LocationCFilesAndCFLAGS = []
 
         # confnode asks for some LDFLAGS
-        if CTNLDFLAGS:
+        LDFLAGS = []
+        if CTNLDFLAGS is not None:
             # LDFLAGS can be either string
-            if isinstance(CTNLDFLAGS, str):
-                LDFLAGS = [CTNLDFLAGS]
+            if isinstance(CTNLDFLAGS, str) or isinstance(CTNLDFLAGS, unicode):
+                LDFLAGS += [CTNLDFLAGS]
             # or list of strings
             elif isinstance(CTNLDFLAGS, list):
-                LDFLAGS = CTNLDFLAGS[:]
-        else:
-            LDFLAGS = []
+                LDFLAGS += CTNLDFLAGS
 
         # recurse through all children, and stack their results
         for CTNChild in self.IECSortedChildren():
@@ -500,7 +499,10 @@
         # Call the OnCloseMethod
         CTNInstance.OnCTNClose()
         # Delete confnode dir
-        shutil.rmtree(CTNInstance.CTNPath())
+        try:
+            shutil.rmtree(CTNInstance.CTNPath())
+        except:
+            pass
         # Remove child of Children
         self.Children[CTNInstance.CTNType].remove(CTNInstance)
         if len(self.Children[CTNInstance.CTNType]) == 0:
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/connectors/PYRO/dialog.py	Wed May 23 20:22:45 2018 +0200
@@ -0,0 +1,69 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+# Copyright (C) 2018: Smarteh
+#
+# See COPYING file for copyrights details.
+
+
+from __future__ import absolute_import
+from __future__ import print_function
+
+import wx
+from controls.UriLocationEditor import IConnectorPanel
+from zope.interface import implementer
+
+URITypes = ["LOCAL", "PYRO", "PYROS"]
+
+
+def PYRO_connector_dialog(confnodesroot):
+    [ID_IPTEXT, ID_PORTTEXT] = [wx.NewId() for _init_ctrls in range(2)]
+
+
+    @implementer(IConnectorPanel)
+    class PYROConnectorPanel(wx.Panel):
+        def __init__(self, typeConnector, parrent, *args, **kwargs):
+            self.type = typeConnector
+            self.parrent = parrent
+            wx.Panel.__init__(self, parrent, *args, **kwargs)
+            self._init_ctrls()
+            self._init_sizers()
+            self.uri = None
+
+        def _init_ctrls(self):
+            self.IpText = wx.TextCtrl(parent=self, id=ID_IPTEXT, size = wx.Size(200, -1))
+            self.PortText = wx.TextCtrl(parent=self, id=ID_PORTTEXT, size = wx.Size(200, -1))
+
+        def _init_sizers(self):
+            self.mainSizer = wx.BoxSizer(wx.VERTICAL)
+            self.uriSizer = wx.BoxSizer(wx.HORIZONTAL)
+            self.portSizer = wx.BoxSizer(wx.HORIZONTAL)
+
+            self.uriSizer.Add(wx.StaticText(self, wx.ID_ANY, "URI host:", size = wx.Size(70, -1)), proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
+            self.uriSizer.AddSpacer((0,0))
+            self.uriSizer.Add(self.IpText, proportion=1, flag=wx.ALIGN_RIGHT)
+            self.mainSizer.Add(self.uriSizer, border=2, flag=wx.ALL)
+
+            self.portSizer.Add(wx.StaticText(self, wx.ID_ANY, "URI port:", size = wx.Size(70, -1)), proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
+            self.portSizer.AddSpacer((0,0))
+            self.portSizer.Add(self.PortText, proportion=1, flag=wx.ALIGN_RIGHT)
+            self.mainSizer.Add(self.portSizer, border=2, flag=wx.ALL)
+
+            self.SetSizer(self.mainSizer)
+
+        def SetURI(self, uri):
+            self.uri = uri
+            uri_list = uri.strip().split(":")
+            length = len(uri_list)
+            if length == 3:
+                self.IpText.SetValue(uri_list[1].strip("/"))
+                self.PortText.SetValue(uri_list[2])
+            elif length == 2:
+                self.IpText.SetValue(uri_list[1].strip("/"))
+
+
+        def GetURI(self):
+            self.uri = self.type+"://"+self.IpText.GetValue()+":"+self.PortText.GetValue()
+            return self.uri
+
+    return PYROConnectorPanel("PYRO", confnodesroot)
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/connectors/WAMP/dialog.py	Wed May 23 20:22:45 2018 +0200
@@ -0,0 +1,98 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+# Copyright (C) 2018: Smarteh
+#
+# See COPYING file for copyrights details.
+
+
+from __future__ import absolute_import
+from __future__ import print_function
+import wx
+from controls.UriLocationEditor import IConnectorPanel
+from zope.interface import implementer
+
+URITypes = ["WAMP", "WAMPS"]
+
+
+def WAMP_connector_dialog(confnodesroot):
+    [ID_IPTEXT, ID_PORTTEXT, ID_REALMTEXT, ID_WAMPIDTEXT, ID_SECURECHECKBOX] = [wx.NewId() for _init_ctrls in range(5)]
+
+
+    @implementer(IConnectorPanel)
+    class WAMPConnectorPanel(wx.Panel):
+        def __init__(self, typeConnector, parrent, *args, **kwargs):
+            self.type = typeConnector
+            self.parrent = parrent
+            wx.Panel.__init__(self, parrent, *args, **kwargs)
+            self._init_ctrls()
+            self._init_sizers()
+            self.uri = None
+
+        def _init_ctrls(self):
+            self.IpText = wx.TextCtrl(parent=self, id=ID_IPTEXT, size = wx.Size(200, -1))
+            self.PortText = wx.TextCtrl(parent=self, id=ID_PORTTEXT, size = wx.Size(200, -1))
+            self.RealmText = wx.TextCtrl(parent=self, id=ID_REALMTEXT, size = wx.Size(200, -1))
+            self.WAMPIDText = wx.TextCtrl(parent=self, id=ID_WAMPIDTEXT, size = wx.Size(200, -1))
+            self.SecureCheckbox = wx.CheckBox(self, ID_SECURECHECKBOX, _("Is connection secure?"))
+
+        def _init_sizers(self):
+            self.mainSizer = wx.BoxSizer(wx.VERTICAL)
+            self.uriSizer = wx.BoxSizer(wx.HORIZONTAL)
+            self.portSizer = wx.BoxSizer(wx.HORIZONTAL)
+            self.realmSizer = wx.BoxSizer(wx.HORIZONTAL)
+            self.wampIDSizer = wx.BoxSizer(wx.HORIZONTAL)
+
+            self.uriSizer.Add(wx.StaticText(self, wx.ID_ANY, _("URI host:"), size = wx.Size(70, -1)), proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
+            self.uriSizer.AddSpacer((0,0))
+            self.uriSizer.Add(self.IpText, proportion=1, flag=wx.ALIGN_RIGHT)
+            self.mainSizer.Add(self.uriSizer, border=2, flag=wx.ALL)
+
+            self.portSizer.Add(wx.StaticText(self, wx.ID_ANY, _("URI port:"), size = wx.Size(70, -1)), proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
+            self.portSizer.AddSpacer((0,0))
+            self.portSizer.Add(self.PortText, proportion=1, flag=wx.ALIGN_RIGHT)
+            self.mainSizer.Add(self.portSizer, border=2, flag=wx.ALL)
+
+            self.realmSizer.Add(wx.StaticText(self, wx.ID_ANY, _("Realm:"), size = wx.Size(70, -1)), proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
+            self.realmSizer.AddSpacer((0, 0))
+            self.realmSizer.Add(self.RealmText, proportion=1, flag=wx.ALIGN_RIGHT)
+            self.mainSizer.Add(self.realmSizer, border=2, flag=wx.ALL)
+
+            self.wampIDSizer.Add(wx.StaticText(self, wx.ID_ANY, _("WAMP ID:"), size = wx.Size(70, -1)), proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
+            self.wampIDSizer.AddSpacer((0, 0))
+            self.wampIDSizer.Add(self.WAMPIDText, proportion=1, flag=wx.ALIGN_RIGHT)
+            self.mainSizer.Add(self.wampIDSizer, border=2, flag=wx.ALL)
+
+            self.mainSizer.Add(self.SecureCheckbox, proportion=1, flag=wx.ALIGN_LEFT)
+
+            self.SetSizer(self.mainSizer)
+
+        def SetURI(self, uri):
+            self.uri = uri
+            uri_list = uri.strip().split(":")
+            length = len(uri_list)
+
+            if length > 0:
+                if uri_list[0] == URITypes[1]:
+                    self.SecureCheckbox.SetValue(True)
+
+                if length > 2:
+                    self.IpText.SetValue(uri_list[1].strip("/"))
+                    wampSett = uri_list[2].split("#")
+                    length2 = len(wampSett)
+                    if length2 > 0:
+                        self.PortText.SetValue(wampSett[0])
+                        if length2 > 1:
+                            self.RealmText.SetValue(wampSett[1])
+                            if length2 > 2:
+                                self.WAMPIDText.SetValue(wampSett[2])
+
+        def GetURI(self):
+            if self.IpText.Validate():
+                typeForURI = self.type + "S" if self.SecureCheckbox.GetValue() else self.type
+                self.uri = typeForURI + "://" + self.IpText.GetValue() + ":" + self.PortText.GetValue() + "#" + self.RealmText.GetValue() + "#" + self.WAMPIDText.GetValue()
+                return self.uri
+            else:
+                return ""
+
+    return WAMPConnectorPanel("WAMP", confnodesroot)
--- a/connectors/__init__.py	Wed May 23 12:13:48 2018 +0300
+++ b/connectors/__init__.py	Wed May 23 20:22:45 2018 +0200
@@ -36,6 +36,12 @@
 def _GetLocalConnectorClassFactory(name):
     return lambda: getattr(__import__(name, globals(), locals()), name + "_connector_factory")
 
+def _GetLocalConnectorClassDialog(name):
+    return lambda: getattr(__import__(name + '.dialog', globals(), locals(), fromlist=['dialog']), name + "_connector_dialog")
+
+def _GetLocalConnectorURITypes(name):
+    return lambda: getattr(__import__(name + '.dialog', globals(), locals(), fromlist=['dialog']), "URITypes", None)
+
 
 connectors = {name:
               _GetLocalConnectorClassFactory(name)
@@ -43,6 +49,12 @@
               if (path.isdir(path.join(_base_path, name)) and
                   not name.startswith("__"))}
 
+connectors_dialog = {name:
+                     {"function":_GetLocalConnectorClassDialog(name), "URITypes": _GetLocalConnectorURITypes(name)}
+                     for name in listdir(_base_path)
+                     if (path.isdir(path.join(_base_path, name)) and
+                         not name.startswith("__"))}
+
 
 def ConnectorFactory(uri, confnodesroot):
     """
@@ -68,3 +80,20 @@
     # import module according to uri type
     connectorclass = connectors[servicetype]()
     return connectorclass(uri, confnodesroot)
+
+def ConnectorDialog(conn_type, confnodesroot):
+    if conn_type not in connectors_dialog:
+        return None
+
+    connectorclass = connectors_dialog[conn_type]["function"]()
+    return connectorclass(confnodesroot)
+
+def GetConnectorFromURI(uri):
+    typeOfConnector = None
+    for conn_type in connectors_dialog:
+        connectorTypes = connectors_dialog[conn_type]["URITypes"]()
+        if connectorTypes and uri in connectorTypes:
+            typeOfConnector = conn_type
+            break
+
+    return typeOfConnector
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/controls/UriLocationEditor.py	Wed May 23 20:22:45 2018 +0200
@@ -0,0 +1,110 @@
+import wx
+from zope.interface import Interface, Attribute
+from zope.interface.verify import verifyObject
+from connectors import connectors_dialog, ConnectorDialog, GetConnectorFromURI
+
+
+[ID_URIWIZARDDIALOG,ID_URITYPECHOICE] = [wx.NewId() for _init_ctrls in range(2)]
+
+class IConnectorPanel(Interface):
+    """This is interface for panel of seperate connector type"""
+    uri = Attribute("""uri of connections""")
+    type = Attribute("""type of connector""")
+
+    def SetURI(uri):
+        """methode for set uri"""
+
+    def GetURI():
+        """metohde for get uri"""
+
+
+class UriLocationEditor(wx.Dialog):
+    def _init_ctrls(self, parent):
+        wx.Dialog.__init__(self, id=ID_URIWIZARDDIALOG,
+              name='UriLocationEditor', parent=parent,
+              title='Uri location')
+        self.UriTypeChoice = wx.Choice(parent=self, id=ID_URIWIZARDDIALOG, choices = self.URITYPES)
+        self.UriTypeChoice.SetSelection(0)
+        self.Bind(wx.EVT_CHOICE, self.OnTypeChoice, self.UriTypeChoice)
+        self.PanelSizer = wx.BoxSizer(wx.HORIZONTAL)
+        self.ButtonSizer = self.CreateButtonSizer(wx.OK|wx.CANCEL)
+
+    def _init_sizers(self):
+        self.mainSizer = wx.BoxSizer(wx.VERTICAL)
+        typeSizer = wx.BoxSizer(wx.HORIZONTAL)
+        typeSizer.Add(wx.StaticText(self,wx.ID_ANY,"URI type:"), border=5, flag=wx.ALIGN_CENTER_VERTICAL|wx.ALL)
+        typeSizer.Add(self.UriTypeChoice, border=5, flag=wx.ALL)
+        self.mainSizer.Add(typeSizer)
+
+        self.mainSizer.Add(self.PanelSizer, border=5, flag=wx.ALL)
+        self.mainSizer.Add(self.ButtonSizer, border=5, flag=wx.BOTTOM|wx.ALIGN_CENTER_HORIZONTAL)
+        self.SetSizer(self.mainSizer)
+
+    def __init__(self, parent, uri):
+        self.URITYPES = ["- Select URI type -"]
+        for connector_type, connector_function in connectors_dialog.iteritems():
+            try:
+                connector_function['function']()
+                self.URITYPES.append(connector_type)
+            except Exception as e:
+                pass
+
+        self.selected = None
+        self.parrent = parent
+        self.logger = self.parrent.CTR.logger
+        self._init_ctrls(parent)
+        self._init_sizers()
+        self.SetURI(uri)
+        self.CenterOnParent()
+
+    def OnTypeChoice(self, event):
+        self._removePanelType()
+        index = event.GetSelection()
+        if index > 0:
+            self.selected = event.GetString()
+            self.panelType = self._getConnectorDialog(self.selected)
+            if self.panelType:
+                self.PanelSizer.Add(self.panelType)
+                self.mainSizer.Layout()
+                self.Fit()
+                self.panelType.Refresh()
+
+    def SetURI(self, uri):
+        self._removePanelType()
+        uri_list = uri.strip().split(":")
+        if uri_list:
+            uri_type = uri_list[0].upper()
+            type = GetConnectorFromURI(uri_type)
+            if type:
+                self.selected = type
+                self.UriTypeChoice.SetStringSelection(self.selected)
+                self.panelType = self._getConnectorDialog(self.selected)
+                if self.panelType:
+                    self.panelType.SetURI(uri)
+                    self.PanelSizer.Add(self.panelType)
+                    self.PanelSizer.Layout()
+                    self.mainSizer.Layout()
+                    self.Fit()
+                    self.panelType.Refresh()
+
+    def GetURI(self):
+        if not self.selected or not self.panelType:
+            return ""
+        else:
+            return self.panelType.GetURI()
+
+    def _removePanelType(self):
+        for i in range(self.PanelSizer.GetItemCount()):
+            item = self.PanelSizer.GetItem(i)
+            item.DeleteWindows()
+            self.PanelSizer.Remove(i)
+            self.Fit()
+        self.PanelSizer.Layout()
+
+    def _getConnectorDialog(self, connectorType):
+        connector = ConnectorDialog(connectorType, self)
+        if connector and IConnectorPanel.providedBy(connector):
+            if verifyObject(IConnectorPanel, connector):
+                return connector
+        else:
+            return None
--- a/editors/ConfTreeNodeEditor.py	Wed May 23 12:13:48 2018 +0300
+++ b/editors/ConfTreeNodeEditor.py	Wed May 23 20:22:45 2018 +0200
@@ -33,7 +33,7 @@
 
 from IDEFrame import TITLE, FILEMENU, PROJECTTREE, PAGETITLES
 
-from controls import TextCtrlAutoComplete
+from controls import TextCtrlAutoComplete, UriLocationEditor
 from dialogs import BrowseValuesLibraryDialog
 from util.BitmapLibrary import GetBitmap
 
@@ -338,6 +338,28 @@
                 msizer.AddWindow(button, flag=wx.ALIGN_CENTER)
         return msizer
 
+    def UriOptions(self, event):
+        CTR = self.ParentWindow.CTR
+        CTR_BeremizRoot = CTR.BeremizRoot
+        CTR_AppFrame = CTR.AppFrame
+
+        # Get connector uri
+        uri = CTR_BeremizRoot.getURI_location().strip()
+        dialog = UriLocationEditor.UriLocationEditor(CTR_AppFrame, uri)
+
+        if dialog.ShowModal() == wx.ID_OK:
+            CTR_BeremizRoot.setURI_location(dialog.GetURI())
+            if CTR._View is not None:
+                CTR._View.RefreshView()
+            if CTR_AppFrame is not None:
+                CTR_AppFrame.RefreshTitle()
+                CTR_AppFrame.RefreshFileMenu()
+                CTR_AppFrame.RefreshEditMenu()
+                CTR_AppFrame.RefreshPageTitles()
+
+        dialog.Destroy()
+
+
     def GenerateSizerElements(self, sizer, elements, path, clean=True):
         if clean:
             sizer.Clear(True)
@@ -484,7 +506,21 @@
                                                         element_path=element_path,
                                                         size=wx.Size(300, -1))
 
-                        boxsizer.AddWindow(textctrl)
+                        if element_infos["name"] == "URI_location":
+                            uriSizer = wx.FlexGridSizer(cols=2, hgap=0, rows=1, vgap=0)
+                            uriSizer.AddGrowableCol(0)
+                            uriSizer.AddGrowableRow(0)
+
+                            self.EditButton = wx.Button(self.ParamsEditor, label='...', size=wx.Size(30, -1))
+                            self.Bind(wx.EVT_BUTTON, self.UriOptions, self.EditButton)
+
+                            uriSizer.AddWindow(textctrl, flag=wx.GROW)
+                            uriSizer.AddWindow(self.EditButton, flag=wx.GROW)
+
+                            boxsizer.AddWindow(uriSizer)
+                        else:
+                            boxsizer.AddWindow(textctrl)
+
                         if element_infos["value"] is not None:
                             textctrl.ChangeValue(str(element_infos["value"]))
                         callback = self.GetTextCtrlCallBackFunction(textctrl, element_path)
--- a/modbus/mb_runtime.c	Wed May 23 12:13:48 2018 +0300
+++ b/modbus/mb_runtime.c	Wed May 23 20:22:45 2018 +0200
@@ -492,12 +492,13 @@
 	for (index=0; index < NUMBER_OF_CLIENT_REQTS; index ++){
 		/*just do the output requests */
 		if (client_requests[index].req_type == req_output){
-			pthread_mutex_lock(&(client_requests[index].coms_buf_mutex));
-			// copy from plcv_buffer to coms_buffer
-			memcpy((void *)client_requests[index].coms_buffer /* destination */,
-			       (void *)client_requests[index].plcv_buffer /* source */,
-			       REQ_BUF_SIZE * sizeof(u16) /* size in bytes */);
-			pthread_mutex_unlock(&(client_requests[index].coms_buf_mutex));
+			if(pthread_mutex_trylock(&(client_requests[index].coms_buf_mutex)) == 0){
+                // copy from plcv_buffer to coms_buffer
+                memcpy((void *)client_requests[index].coms_buffer /* destination */,
+                       (void *)client_requests[index].plcv_buffer /* source */,
+                       REQ_BUF_SIZE * sizeof(u16) /* size in bytes */);
+                pthread_mutex_unlock(&(client_requests[index].coms_buf_mutex));
+            }
 		}
 	}
 }
@@ -512,12 +513,13 @@
 	for (index=0; index < NUMBER_OF_CLIENT_REQTS; index ++){
 		/*just do the input requests */
 		if (client_requests[index].req_type == req_input){
-			pthread_mutex_lock(&(client_requests[index].coms_buf_mutex));
-			// copy from coms_buffer to plcv_buffer
-			memcpy((void *)client_requests[index].plcv_buffer /* destination */,
-			       (void *)client_requests[index].coms_buffer /* source */,
-			       REQ_BUF_SIZE * sizeof(u16) /* size in bytes */);
-			pthread_mutex_unlock(&(client_requests[index].coms_buf_mutex));
+			if(pthread_mutex_trylock(&(client_requests[index].coms_buf_mutex)) == 0){
+                // copy from coms_buffer to plcv_buffer
+                memcpy((void *)client_requests[index].plcv_buffer /* destination */,
+                       (void *)client_requests[index].coms_buffer /* source */,
+                       REQ_BUF_SIZE * sizeof(u16) /* size in bytes */);
+                pthread_mutex_unlock(&(client_requests[index].coms_buf_mutex));
+            }
 		}
 	}
 }