add processRecord
This commit is contained in:
@ -14,6 +14,7 @@ INC += pv/channelProviderLocal.h
|
|||||||
INC += pv/pvDatabase.h
|
INC += pv/pvDatabase.h
|
||||||
INC += pv/traceRecord.h
|
INC += pv/traceRecord.h
|
||||||
INC += pv/removeRecord.h
|
INC += pv/removeRecord.h
|
||||||
|
INC += pv/processRecord.h
|
||||||
INC += pv/pvSupport.h
|
INC += pv/pvSupport.h
|
||||||
INC += pv/controlSupport.h
|
INC += pv/controlSupport.h
|
||||||
INC += pv/scalarAlarmSupport.h
|
INC += pv/scalarAlarmSupport.h
|
||||||
|
@ -68,7 +68,7 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual void reset();
|
virtual void reset();
|
||||||
static ControlSupportPtr create(PVRecordPtr const & pvRecord);
|
static ControlSupportPtr create(PVRecordPtr const & pvRecord);
|
||||||
static epics::pvData::StructureConstPtr controlField();
|
static epics::pvData::StructureConstPtr controlField(epics::pvData::ScalarType scalarType);
|
||||||
private:
|
private:
|
||||||
ControlSupport(PVRecordPtr const & pvRecord);
|
ControlSupport(PVRecordPtr const & pvRecord);
|
||||||
PVRecordPtr pvRecord;
|
PVRecordPtr pvRecord;
|
||||||
@ -77,8 +77,7 @@ private:
|
|||||||
epics::pvData::PVDoublePtr pvLimitLow;
|
epics::pvData::PVDoublePtr pvLimitLow;
|
||||||
epics::pvData::PVDoublePtr pvLimitHigh;
|
epics::pvData::PVDoublePtr pvLimitHigh;
|
||||||
epics::pvData::PVDoublePtr pvMinStep;
|
epics::pvData::PVDoublePtr pvMinStep;
|
||||||
epics::pvData::PVDoublePtr pvOutputValue;
|
epics::pvData::PVScalarPtr pvOutputValue;
|
||||||
double requestedValue;
|
|
||||||
double currentValue;
|
double currentValue;
|
||||||
bool isMinStep;
|
bool isMinStep;
|
||||||
};
|
};
|
||||||
|
79
src/pv/processRecord.h
Normal file
79
src/pv/processRecord.h
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
/* processRecord.h */
|
||||||
|
/**
|
||||||
|
* Copyright - See the COPYRIGHT that is included with this distribution.
|
||||||
|
* EPICS pvData is distributed subject to a Software License Agreement found
|
||||||
|
* in file LICENSE that is included with this distribution.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* @author mrk
|
||||||
|
* @date 2013.04.18
|
||||||
|
*/
|
||||||
|
#ifndef PROCESSRECORD_H
|
||||||
|
#define PROCESSRECORD_H
|
||||||
|
|
||||||
|
#include <map>
|
||||||
|
#include <epicsThread.h>
|
||||||
|
#include <pv/event.h>
|
||||||
|
#include <pv/channelProviderLocal.h>
|
||||||
|
#include <shareLib.h>
|
||||||
|
|
||||||
|
namespace epics { namespace pvDatabase {
|
||||||
|
|
||||||
|
typedef std::tr1::shared_ptr<epicsThread> EpicsThreadPtr;
|
||||||
|
|
||||||
|
class ProcessRecord;
|
||||||
|
typedef std::tr1::shared_ptr<ProcessRecord> ProcessRecordPtr;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Process another record in the same database.
|
||||||
|
*
|
||||||
|
* A record to process another record
|
||||||
|
* It is meant to be used via a channelPutGet request.
|
||||||
|
* The argument has one field: recordName.
|
||||||
|
* The result has a field named status.
|
||||||
|
*/
|
||||||
|
class epicsShareClass ProcessRecord :
|
||||||
|
public PVRecord,
|
||||||
|
public epicsThreadRunable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
POINTER_DEFINITIONS(ProcessRecord);
|
||||||
|
/**
|
||||||
|
* Factory methods to create ProcessRecord.
|
||||||
|
* @param recordName The name for the ProcessRecord.
|
||||||
|
* @param delay Delay time to wait between process requests.
|
||||||
|
* @return A shared pointer to ProcessRecord.
|
||||||
|
*/
|
||||||
|
static ProcessRecordPtr create(
|
||||||
|
std::string const & recordName,double delay);
|
||||||
|
/**
|
||||||
|
* standard init method required by PVRecord
|
||||||
|
* @return true unless record name already exists.
|
||||||
|
*/
|
||||||
|
virtual bool init();
|
||||||
|
/**
|
||||||
|
* @brief Process the record specified by recordName.
|
||||||
|
*/
|
||||||
|
virtual void process();
|
||||||
|
virtual void run();
|
||||||
|
void startThread();
|
||||||
|
void stop();
|
||||||
|
private:
|
||||||
|
ProcessRecord(
|
||||||
|
std::string const & recordName,
|
||||||
|
epics::pvData::PVStructurePtr const & pvStructure,double delay);
|
||||||
|
double delay;
|
||||||
|
EpicsThreadPtr thread;
|
||||||
|
epics::pvData::Event runStop;
|
||||||
|
epics::pvData::Event runReturn;
|
||||||
|
PVDatabasePtr pvDatabase;
|
||||||
|
PVRecordMap pvRecordMap;
|
||||||
|
epics::pvData::PVStringPtr pvCommand;
|
||||||
|
epics::pvData::PVStringPtr pvRecordName;
|
||||||
|
epics::pvData::PVStringPtr pvResult;
|
||||||
|
epics::pvData::Mutex mutex;
|
||||||
|
};
|
||||||
|
|
||||||
|
}}
|
||||||
|
|
||||||
|
#endif /* PROCESSRECORD_H */
|
@ -4,9 +4,12 @@ SRC_DIRS += $(PVDATABASE_SRC)/special
|
|||||||
|
|
||||||
LIBSRCS += traceRecord.cpp
|
LIBSRCS += traceRecord.cpp
|
||||||
LIBSRCS += removeRecord.cpp
|
LIBSRCS += removeRecord.cpp
|
||||||
|
LIBSRCS += processRecord.cpp
|
||||||
|
|
||||||
DBD += traceRecordRegister.dbd
|
DBD += traceRecordRegister.dbd
|
||||||
DBD += removeRecordRegister.dbd
|
DBD += removeRecordRegister.dbd
|
||||||
|
DBD += processRecordRegister.dbd
|
||||||
|
|
||||||
LIBSRCS += traceRecordRegister.cpp
|
LIBSRCS += traceRecordRegister.cpp
|
||||||
LIBSRCS += removeRecordRegister.cpp
|
LIBSRCS += removeRecordRegister.cpp
|
||||||
|
LIBSRCS += processRecordRegister.cpp
|
||||||
|
146
src/special/processRecord.cpp
Normal file
146
src/special/processRecord.cpp
Normal file
@ -0,0 +1,146 @@
|
|||||||
|
/* processRecord.cpp */
|
||||||
|
/**
|
||||||
|
* Copyright - See the COPYRIGHT that is included with this distribution.
|
||||||
|
* EPICS pvData is distributed subject to a Software License Agreement found
|
||||||
|
* in file LICENSE that is included with this distribution.
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* @author mrk
|
||||||
|
* @date 2013.04.18
|
||||||
|
*/
|
||||||
|
#include <map>
|
||||||
|
#include <epicsThread.h>
|
||||||
|
#include <pv/event.h>
|
||||||
|
#include <pv/channelProviderLocal.h>
|
||||||
|
#include <shareLib.h>
|
||||||
|
|
||||||
|
#define epicsExportSharedSymbols
|
||||||
|
#include <pv/processRecord.h>
|
||||||
|
|
||||||
|
using std::tr1::static_pointer_cast;
|
||||||
|
using namespace epics::pvData;
|
||||||
|
using namespace epics::pvAccess;
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
namespace epics { namespace pvDatabase {
|
||||||
|
|
||||||
|
ProcessRecordPtr ProcessRecord::create(
|
||||||
|
std::string const & recordName,double delay)
|
||||||
|
{
|
||||||
|
FieldCreatePtr fieldCreate = getFieldCreate();
|
||||||
|
PVDataCreatePtr pvDataCreate = getPVDataCreate();
|
||||||
|
StructureConstPtr topStructure = fieldCreate->createFieldBuilder()->
|
||||||
|
addNestedStructure("argument")->
|
||||||
|
add("command",pvString)->
|
||||||
|
add("recordName",pvString)->
|
||||||
|
endNested()->
|
||||||
|
addNestedStructure("result") ->
|
||||||
|
add("status",pvString) ->
|
||||||
|
endNested()->
|
||||||
|
createStructure();
|
||||||
|
PVStructurePtr pvStructure = pvDataCreate->createPVStructure(topStructure);
|
||||||
|
ProcessRecordPtr pvRecord(
|
||||||
|
new ProcessRecord(recordName,pvStructure,delay));
|
||||||
|
if(!pvRecord->init()) pvRecord.reset();
|
||||||
|
return pvRecord;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProcessRecord::startThread()
|
||||||
|
{
|
||||||
|
thread = EpicsThreadPtr(new epicsThread(
|
||||||
|
*this,
|
||||||
|
"processRecord",
|
||||||
|
epicsThreadGetStackSize(epicsThreadStackSmall),
|
||||||
|
epicsThreadPriorityLow));
|
||||||
|
thread->start();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProcessRecord::stop()
|
||||||
|
{
|
||||||
|
runStop.signal();
|
||||||
|
runReturn.wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ProcessRecord::ProcessRecord(
|
||||||
|
std::string const & recordName,
|
||||||
|
epics::pvData::PVStructurePtr const & pvStructure,double delay)
|
||||||
|
: PVRecord(recordName,pvStructure),
|
||||||
|
delay(delay),
|
||||||
|
pvDatabase(PVDatabase::getMaster())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ProcessRecord::init()
|
||||||
|
{
|
||||||
|
initPVRecord();
|
||||||
|
PVStructurePtr pvStructure = getPVStructure();
|
||||||
|
pvCommand = pvStructure->getSubField<PVString>("argument.command");
|
||||||
|
pvRecordName = pvStructure->getSubField<PVString>("argument.recordName");
|
||||||
|
if(!pvRecordName) return false;
|
||||||
|
pvResult = pvStructure->getSubField<PVString>("result.status");
|
||||||
|
if(!pvResult) return false;
|
||||||
|
startThread();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProcessRecord::process()
|
||||||
|
{
|
||||||
|
string recordName = pvRecordName->get();
|
||||||
|
string command = pvCommand->get();
|
||||||
|
if(command.compare("add")==0) {
|
||||||
|
epicsGuard<epics::pvData::Mutex> guard(mutex);
|
||||||
|
std::map<std::string,PVRecordPtr>::iterator iter = pvRecordMap.find(recordName);
|
||||||
|
if(iter!=pvRecordMap.end()) {
|
||||||
|
pvResult->put(recordName + " already pesent");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
PVRecordPtr pvRecord = pvDatabase->findRecord(recordName);
|
||||||
|
if(!pvRecord) {
|
||||||
|
pvResult->put(recordName + " not in pvDatabase");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
pvRecordMap.insert(PVRecordMap::value_type(recordName,pvRecord));
|
||||||
|
pvResult->put("success");
|
||||||
|
return;
|
||||||
|
} else if(command.compare("remove")==0) {
|
||||||
|
epicsGuard<epics::pvData::Mutex> guard(mutex);
|
||||||
|
std::map<std::string,PVRecordPtr>::iterator iter = pvRecordMap.find(recordName);
|
||||||
|
if(iter==pvRecordMap.end()) {
|
||||||
|
pvResult->put(recordName + " not found");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
PVRecordPtr pvRecord = (*iter).second;
|
||||||
|
pvDatabase->removeRecord(pvRecord);
|
||||||
|
pvResult->put("success");
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
pvResult->put(command + " not a valid command: only add and remove are valid");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProcessRecord::run()
|
||||||
|
{
|
||||||
|
while(true) {
|
||||||
|
if(runStop.tryWait()) {
|
||||||
|
runReturn.signal();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(delay>0.0) epicsThreadSleep(delay);
|
||||||
|
epicsGuard<epics::pvData::Mutex> guard(mutex);
|
||||||
|
PVRecordMap::iterator iter;
|
||||||
|
for(iter = pvRecordMap.begin(); iter!=pvRecordMap.end(); ++iter) {
|
||||||
|
PVRecordPtr pvRecord = (*iter).second;
|
||||||
|
pvRecord->lock();
|
||||||
|
pvRecord->beginGroupPut();
|
||||||
|
pvRecord->process();
|
||||||
|
pvRecord->endGroupPut();
|
||||||
|
pvRecord->unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}}
|
||||||
|
|
72
src/special/processRecordRegister.cpp
Normal file
72
src/special/processRecordRegister.cpp
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
/*
|
||||||
|
* Copyright information and license terms for this software can be
|
||||||
|
* found in the file LICENSE that is included with the distribution
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author mrk
|
||||||
|
* @date 2013.07.24
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/* Author: Marty Kraimer */
|
||||||
|
|
||||||
|
#include <cstddef>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstddef>
|
||||||
|
#include <string>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include <cantProceed.h>
|
||||||
|
#include <epicsStdio.h>
|
||||||
|
#include <epicsMutex.h>
|
||||||
|
#include <epicsEvent.h>
|
||||||
|
#include <epicsThread.h>
|
||||||
|
#include <iocsh.h>
|
||||||
|
|
||||||
|
#include <pv/pvIntrospect.h>
|
||||||
|
#include <pv/pvData.h>
|
||||||
|
#include <pv/pvAccess.h>
|
||||||
|
#include <pv/pvDatabase.h>
|
||||||
|
|
||||||
|
#include <epicsExport.h>
|
||||||
|
#include <pv/processRecord.h>
|
||||||
|
|
||||||
|
using namespace epics::pvData;
|
||||||
|
using namespace epics::pvAccess;
|
||||||
|
using namespace epics::pvDatabase;
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
static const iocshArg testArg0 = { "recordName", iocshArgString };
|
||||||
|
static const iocshArg testArg1 = { "delay", iocshArgDouble };
|
||||||
|
static const iocshArg *testArgs[] = {
|
||||||
|
&testArg0,&testArg1};
|
||||||
|
|
||||||
|
static const iocshFuncDef processRecordFuncDef = {"processRecordCreate", 2,testArgs};
|
||||||
|
|
||||||
|
static void processRecordCallFunc(const iocshArgBuf *args)
|
||||||
|
{
|
||||||
|
char *recordName = args[0].sval;
|
||||||
|
if(!recordName) {
|
||||||
|
throw std::runtime_error("processRecordCreate invalid number of arguments");
|
||||||
|
}
|
||||||
|
double delay = args[1].dval;
|
||||||
|
if(delay<0.0) delay = 1.0;
|
||||||
|
ProcessRecordPtr record = ProcessRecord::create(recordName,delay);
|
||||||
|
bool result = PVDatabase::getMaster()->addRecord(record);
|
||||||
|
if(!result) cout << "recordname" << " not added" << endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void processRecordRegister(void)
|
||||||
|
{
|
||||||
|
static int firstTime = 1;
|
||||||
|
if (firstTime) {
|
||||||
|
firstTime = 0;
|
||||||
|
iocshRegister(&processRecordFuncDef, processRecordCallFunc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
epicsExportRegistrar(processRecordRegister);
|
||||||
|
}
|
1
src/special/processRecordRegister.dbd
Normal file
1
src/special/processRecordRegister.dbd
Normal file
@ -0,0 +1 @@
|
|||||||
|
registrar("processRecordRegister")
|
@ -8,8 +8,10 @@
|
|||||||
* @author mrk
|
* @author mrk
|
||||||
* @date 2013.04.18
|
* @date 2013.04.18
|
||||||
*/
|
*/
|
||||||
#define epicsExportSharedSymbols
|
#include <shareLib.h>
|
||||||
|
#include <pv/channelProviderLocal.h>
|
||||||
|
|
||||||
|
#define epicsExportSharedSymbols
|
||||||
#include <pv/removeRecord.h>
|
#include <pv/removeRecord.h>
|
||||||
|
|
||||||
using std::tr1::static_pointer_cast;
|
using std::tr1::static_pointer_cast;
|
||||||
|
@ -8,8 +8,10 @@
|
|||||||
* @author mrk
|
* @author mrk
|
||||||
* @date 2013.04.18
|
* @date 2013.04.18
|
||||||
*/
|
*/
|
||||||
#define epicsExportSharedSymbols
|
#include <shareLib.h>
|
||||||
|
#include <pv/channelProviderLocal.h>
|
||||||
|
|
||||||
|
#define epicsExportSharedSymbols
|
||||||
#include <pv/traceRecord.h>
|
#include <pv/traceRecord.h>
|
||||||
|
|
||||||
using std::tr1::static_pointer_cast;
|
using std::tr1::static_pointer_cast;
|
||||||
|
@ -29,14 +29,14 @@ ControlSupport::~ControlSupport()
|
|||||||
cout << "ControlSupport::~ControlSupport()\n";
|
cout << "ControlSupport::~ControlSupport()\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
epics::pvData::StructureConstPtr ControlSupport::controlField()
|
epics::pvData::StructureConstPtr ControlSupport::controlField(ScalarType scalarType)
|
||||||
{
|
{
|
||||||
return FieldBuilder::begin()
|
return FieldBuilder::begin()
|
||||||
->setId("control_t")
|
->setId("control_t")
|
||||||
->add("limitLow", pvDouble)
|
->add("limitLow", pvDouble)
|
||||||
->add("limitHigh", pvDouble)
|
->add("limitHigh", pvDouble)
|
||||||
->add("minStep", pvDouble)
|
->add("minStep", pvDouble)
|
||||||
->add("outputValue", pvDouble)
|
->add("outputValue", scalarType)
|
||||||
->createStructure();
|
->createStructure();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -71,7 +71,7 @@ bool ControlSupport::init(PVFieldPtr const & pv,PVFieldPtr const & pvsup)
|
|||||||
pvLimitLow = pvControl->getSubField<PVDouble>("limitLow");
|
pvLimitLow = pvControl->getSubField<PVDouble>("limitLow");
|
||||||
pvLimitHigh = pvControl->getSubField<PVDouble>("limitHigh");
|
pvLimitHigh = pvControl->getSubField<PVDouble>("limitHigh");
|
||||||
pvMinStep = pvControl->getSubField<PVDouble>("minStep");
|
pvMinStep = pvControl->getSubField<PVDouble>("minStep");
|
||||||
pvOutputValue = pvControl->getSubField<PVDouble>("outputValue");
|
pvOutputValue = pvControl->getSubField<PVScalar>("outputValue");
|
||||||
}
|
}
|
||||||
if(!pvControl || !pvLimitLow || !pvLimitHigh || !pvMinStep || !pvOutputValue) {
|
if(!pvControl || !pvLimitLow || !pvLimitHigh || !pvMinStep || !pvOutputValue) {
|
||||||
cout << "ControlSupport for record " << pvRecord->getRecordName()
|
cout << "ControlSupport for record " << pvRecord->getRecordName()
|
||||||
@ -79,8 +79,7 @@ bool ControlSupport::init(PVFieldPtr const & pv,PVFieldPtr const & pvsup)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
ConvertPtr convert = getConvert();
|
ConvertPtr convert = getConvert();
|
||||||
requestedValue = convert->toDouble(pvValue);
|
currentValue = convert->toDouble(pvValue);
|
||||||
currentValue = requestedValue;
|
|
||||||
isMinStep = false;
|
isMinStep = false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -89,38 +88,37 @@ bool ControlSupport::process()
|
|||||||
{
|
{
|
||||||
ConvertPtr convert = getConvert();
|
ConvertPtr convert = getConvert();
|
||||||
double value = convert->toDouble(pvValue);
|
double value = convert->toDouble(pvValue);
|
||||||
if(value==requestedValue&&value==currentValue) return false;
|
if(!isMinStep && value==currentValue) return false;
|
||||||
if(!isMinStep) requestedValue = value;
|
|
||||||
double limitLow = pvLimitLow->get();
|
double limitLow = pvLimitLow->get();
|
||||||
double limitHigh = pvLimitHigh->get();
|
double limitHigh = pvLimitHigh->get();
|
||||||
double minStep = pvMinStep->get();
|
double minStep = pvMinStep->get();
|
||||||
if(limitHigh>limitLow) {
|
if(limitHigh>limitLow) {
|
||||||
if(requestedValue>limitHigh) requestedValue = limitHigh;
|
if(value>limitHigh) value = limitHigh;
|
||||||
if(requestedValue<limitLow) requestedValue = limitLow;
|
if(value<limitLow) value = limitLow;
|
||||||
}
|
}
|
||||||
double diff = requestedValue - currentValue;
|
double diff = value - currentValue;
|
||||||
double outputValue = requestedValue;
|
double outputValue = value;
|
||||||
if(minStep>0.0) {
|
if(minStep>0.0) {
|
||||||
if(diff<0.0) {
|
if(diff<0.0) {
|
||||||
outputValue = currentValue - minStep;
|
outputValue = currentValue - minStep;
|
||||||
isMinStep = true;
|
isMinStep = true;
|
||||||
if(outputValue<requestedValue) {
|
if(outputValue<value) {
|
||||||
outputValue = requestedValue;
|
outputValue = value;
|
||||||
isMinStep = false;
|
isMinStep = false;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
outputValue = currentValue + minStep;
|
outputValue = currentValue + minStep;
|
||||||
isMinStep = true;
|
isMinStep = true;
|
||||||
if(outputValue>requestedValue) {
|
if(outputValue>value) {
|
||||||
outputValue = requestedValue;
|
outputValue = value;
|
||||||
isMinStep = false;
|
isMinStep = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
currentValue = outputValue;
|
currentValue = outputValue;
|
||||||
pvOutputValue->put(outputValue);
|
convert->fromDouble(pvOutputValue,outputValue);
|
||||||
if(!isMinStep && (outputValue!=requestedValue)) {
|
if(!isMinStep && (outputValue!=value)) {
|
||||||
convert->fromDouble(pvValue,requestedValue);
|
convert->fromDouble(pvValue,value);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user