SourceCode/Bond/Servo/HsmsPassive.cpp
@@ -665,6 +665,27 @@
         maxId = item->getVarialbleId();
      }
   }
   for (auto item : m_dataVariabels) {
      if (item && item->getVarialbleId() > maxId) {
         maxId = item->getVarialbleId();
      }
   }
   return maxId;
}
unsigned int CHsmsPassive::getMaxDataVariableId() const
{
   unsigned int maxId = 0;
   for (auto item : m_variabels) {
      if (item && item->getVarialbleId() > maxId) {
         maxId = item->getVarialbleId();
      }
   }
   for (auto item : m_dataVariabels) {
      if (item && item->getVarialbleId() > maxId) {
         maxId = item->getVarialbleId();
      }
   }
   return maxId;
}
@@ -895,6 +916,75 @@
   if (filepath.empty()) return -3;
   return writeVariablesToFile(filepath);
}
int CHsmsPassive::deleteDataVariable(int dvid)
{
   Lock();
   auto it = std::find_if(m_dataVariabels.begin(), m_dataVariabels.end(), [=](SERVO::CDataVariable* v) {
      return v != nullptr && v->getVarialbleId() == (unsigned int)dvid;
      });
   if (it == m_dataVariabels.end()) {
      Unlock();
      return -1;
   }
   delete *it;
   m_dataVariabels.erase(it);
   auto filepath = m_strDataVariableFilepath;
   Unlock();
   if (filepath.empty()) return -2;
   return writeDataVariablesToFile(filepath);
}
int CHsmsPassive::addDataVariable(const char* pszName, const char* pszFormat, const char* pszRemark, int& outId)
{
   if (pszName == nullptr || pszFormat == nullptr) return -1;
   std::string fmt = pszFormat;
   std::transform(fmt.begin(), fmt.end(), fmt.begin(), ::toupper);
   if (!isValidFormat(fmt)) return -2;
   Lock();
   int maxId = 0;
   for (auto v : m_dataVariabels) {
      if (v != nullptr && static_cast<int>(v->getVarialbleId()) > maxId) {
         maxId = static_cast<int>(v->getVarialbleId());
      }
   }
   outId = maxId + 1;
   SERVO::CDataVariable* pNew = new SERVO::CDataVariable(std::to_string(outId).c_str(), pszName, fmt.c_str(), pszRemark ? pszRemark : "");
   m_dataVariabels.push_back(pNew);
   auto filepath = m_strDataVariableFilepath;
   Unlock();
   if (filepath.empty()) return -3;
   return writeDataVariablesToFile(filepath);
}
int CHsmsPassive::updateDataVariable(int dvid, const char* pszName, const char* pszFormat, const char* pszRemark)
{
   if (pszName == nullptr || pszFormat == nullptr) return -1;
   std::string fmt = pszFormat;
   std::transform(fmt.begin(), fmt.end(), fmt.begin(), ::toupper);
   if (!isValidFormat(fmt)) return -2;
   Lock();
   auto it = std::find_if(m_dataVariabels.begin(), m_dataVariabels.end(), [=](SERVO::CDataVariable* v) {
      return v != nullptr && v->getVarialbleId() == (unsigned int)dvid;
      });
   if (it == m_dataVariabels.end()) {
      Unlock();
      return -4;
   }
   (*it)->setName(pszName);
   (*it)->setFormat(fmt.c_str());
   (*it)->setRemark(pszRemark ? pszRemark : "");
   auto filepath = m_strDataVariableFilepath;
   Unlock();
   if (filepath.empty()) return -3;
   return writeDataVariablesToFile(filepath);
}
int CHsmsPassive::writeVariablesToFile(const std::string& filepath)
@@ -3315,6 +3405,26 @@
   // 解释数据,得到CProcessJob
   // 容量前置检查:当前实现仅支持单批 PJ 集合,如果已有 PJ/CJ,直接返回 ACKA=false
   if (m_pModel != nullptr && !m_pModel->getMaster().isProcessJobsEmpty()) {
      IMessage* pMessage = NULL;
      HSMS_Create1Message(pMessage, m_nSessionId, 16, 16, ++m_nSystemByte);
      ASSERT(pMessage);
      pMessage->getBody()->addItem(); // PRJOBID list 为空
      ISECS2Item* pItemErrors = pMessage->getBody()->addItem();
      pItemErrors->addBoolItem(false, "ACKA");
      ISECS2Item* pItemErrors2 = pItemErrors->addItem();
      auto err = pItemErrors2->addItem();
      err->addU4Item(1000, "ERRCODE");
      err->addItem("PJobSpace=0 (existing ProcessJob/ControlJob)", "ERRTEXT");
      m_pPassive->sendMessage(pMessage);
      LOGI("<HSMS>[SEND]sessionId:%d, sType:%d systemBytes:%d",
         pMessage->getHeader()->sessionId, pMessage->getHeader()->sType, pMessage->getHeader()->systemBytes);
      LogSecsMessageBrief("<HSMS>[SEND]", pMessage);
      HSMS_Destroy1Message(pMessage);
      return 0;
   }
   ISECS2Item* pItemPjs, * pItemPj,* pItemCarriers, * pItemCarrier, *pItemSlots, *pItemRecipes;
   unsigned short DATAID;
   const char* pszPrjobid, *pszMF, *pszCarrierId, *pszRecipeName;
@@ -3366,7 +3476,11 @@
   }
   // 回复报文
   ASSERT(m_listener.onPRJobMultiCreate != nullptr);
   int nRet = m_listener.onPRJobMultiCreate(this, pjs);
   // 回复报文(在校验/落库后再回复,以便带上真实的 issues)
   IMessage* pMessage = NULL;
   HSMS_Create1Message(pMessage, m_nSessionId, 16, 16, ++m_nSystemByte);
   ASSERT(pMessage);
@@ -3392,15 +3506,15 @@
         }
      }
   }
   else {
      pItemErrors->addBoolItem(true, "ACKA");
      pItemErrors->addItem(); // 空列表
   }
   m_pPassive->sendMessage(pMessage);
   LOGI("<HSMS>[SEND]sessionId:%d, sType:%d systemBytes:%d",
      pMessage->getHeader()->sessionId, pMessage->getHeader()->sType, pMessage->getHeader()->systemBytes);
   LogSecsMessageBrief("<HSMS>[SEND]", pMessage);
   HSMS_Destroy1Message(pMessage);
   ASSERT(m_listener.onPRJobMultiCreate != nullptr);
   int nRet = m_listener.onPRJobMultiCreate(this, pjs);
   // 释放有问题(未添加到master)的内存
@@ -3455,6 +3569,10 @@
   if (pEvent == nullptr) {
      return ER_NO_EVENT;
   }
   // 触发 PauseEvent 检测桩(由 Master 负责实际策略)
   if (m_pModel != nullptr) {
      m_pModel->getMaster().handleCollectionEvent(CEID);
   }
   SERVO::CReport* pReport = pEvent->getFirstReport();