Differences
This shows you the differences between two versions of the page.
Both sides previous revision Previous revision Next revision | Previous revision | ||
scripting_for_plc_servers [2020/09/07 12:15] wikiadmin |
scripting_for_plc_servers [2021/09/10 16:09] (current) wikiadmin |
||
---|---|---|---|
Line 1: | Line 1: | ||
- | ==== TransSECS Devices - Scripting for PLC Servers ==== | + | ===== TransSECS Devices - Scripting for PLC Servers |
To write a value to a server in the devices the format is: | To write a value to a server in the devices the format is: | ||
Line 30: | Line 30: | ||
</ | </ | ||
- | **Set a value to a field in a message and send the message** | + | ==== Set a value to a field in a message and send the message |
Line 38: | Line 38: | ||
// find the message | // find the message | ||
- | var message = secsInterface.getMessageBean(" | + | var message = secsInterface.getMessageBean(" |
- | message.setElementName(15); // "ElementName" is the name you gave the element in TransSECS | + | message.setSlotNumber(15); // "SlotNumber" is the name you gave to an element in TransSECS |
message.sendMessage(); | message.sendMessage(); | ||
</ | </ | ||
- | Here's a more concreted | + | Here's a more concrete |
<code javascript> | <code javascript> | ||
Line 65: | Line 65: | ||
</ | </ | ||
- | **Register for Connection Status Notification in TransSECS Host Application** | + | ==== Register for Connection Status Notification in TransSECS Host Application |
In a GEMHost application the TransSECS controller publishes the status of the connection. | In a GEMHost application the TransSECS controller publishes the status of the connection. | ||
Line 98: | Line 98: | ||
The details of the connection status are [[programmatic_host_connection_status|described here]] | The details of the connection status are [[programmatic_host_connection_status|described here]] | ||
- | **Sending | + | ==== Writing |
If a message is received from the host contains a list of variable length it's easy to split this list and place it in registers on the PLC. | If a message is received from the host contains a list of variable length it's easy to split this list and place it in registers on the PLC. | ||
Line 134: | Line 134: | ||
</ | </ | ||
+ | ==== Reading a Variable Length List from a PLC and updating a VID ==== | ||
+ | |||
+ | The most challenging part of reading the list from the PLC is managing the length of the list. | ||
+ | |||
+ | Here we trigger the script on the variable that is the list length (ListVIDLengthFromPLC) | ||
+ | |||
+ | {{: | ||
+ | |||
+ | and use that script to populate a VID " | ||
+ | |||
+ | {{: | ||
+ | |||
+ | The register VIDListArray is used to read the elements of the list individually. | ||
+ | |||
+ | After reading the list from the PLC and setting the VID, the script resets the length variable in the PLC ready for the next update of the variable. | ||
+ | |||
+ | <code javascript> | ||
+ | var TransSecsController = Java.type(" | ||
+ | var SecsFormat20 = Java.type(" | ||
+ | var SecsFormat00 = Java.type(" | ||
+ | var LongValueObject = Java.type(" | ||
+ | |||
+ | |||
+ | listLength = incomingValue.getIntValue(); | ||
+ | //print ("List Length "+ listLength); | ||
+ | try { | ||
+ | if ( listLength > 0 ) { // will reset this to zero after update | ||
+ | |||
+ | print ("List Length "+ listLength); | ||
+ | secsInterface=TransSecsController.findController(" | ||
+ | gh = secsInterface.getGemHandler(); | ||
+ | |||
+ | listValue = new SecsFormat00(); | ||
+ | |||
+ | for (id =0 ; id < listLength ; id++ ) { | ||
+ | / | ||
+ | / | ||
+ | incomingString = / | ||
+ | listValue.add(new SecsFormat20(incomingString)); | ||
+ | } | ||
+ | print ("List "+ listValue); | ||
+ | gh.setValue(" | ||
+ | |||
+ | / | ||
+ | } | ||
+ | } catch ( e ) { | ||
+ | print (" | ||
+ | } | ||
+ | 0; // update the server with a zero... | ||
+ | </ | ||
+ | |||
+ | If the PLC Device supports an Array type, you can use this value directly in the Device Name/Tag Name dropdown. | ||
+ | |||
+ | For a ModbusArray, | ||
+ | |||
+ | {{: | ||
+ | |||
+ | Unsigned Word - SecsFormat52 | ||
+ | Signed Word - SecsFormat54 | ||
+ | Unsigned Double Word - SecsFormat50 | ||
+ | Signed Double Word - SecsFormat50 | ||
+ | Float - SecsFormat44 | ||
+ | Coil - SecsFormat11 | ||
+ | Discrete - SecsFormat11 | ||
+ | |||
+ | For an S7 Array the conversions X,B,W,I,D all convert to SecsFormat5. R (real) convert to SecsFormat44 and S (String) to SecsFormat20. | ||
+ | |||
+ | For improved control over the conversion, the SecsFormat00 can be created in a script. | ||
+ | |||
+ | In this example we create a SecsFormat00 containing signed integers | ||
+ | <code javascript> | ||
+ | var TransSecsController = Java.type(" | ||
+ | var SecsFormat34 = Java.type(" | ||
+ | var SecsFormatValueObject = Java.type(" | ||
+ | var SecsFormat00 = Java.type(" | ||
+ | |||
+ | listValue = new SecsFormat00(); | ||
+ | try { | ||
+ | |||
+ | secsInterface=TransSecsController.findController(" | ||
+ | gh = secsInterface.getGemHandler(); | ||
+ | |||
+ | for (id =0 ; id < incomingValue.size() ; id++ ) { | ||
+ | listValue.add(new SecsFormat34(incomingValue.get(0).getIntValue())); | ||
+ | } | ||
+ | print ("List "+ listValue); | ||
+ | gh.setValue(" | ||
+ | |||
+ | } | ||
+ | } catch ( e ) { | ||
+ | print (" | ||
+ | } | ||
+ | new SecsFormatValueObject(listValue); | ||
+ | </ | ||
+ | |||
+ | ==== Reading a File and Storing as Binary Data in the PLC ==== | ||
+ | |||
+ | This example uses the Melsec Binary (MelsecBinary) server. | ||
+ | |||
+ | |||
+ | < | ||
+ | var Deflater = Java.type(" | ||
+ | var Inflater = Java.type(" | ||
+ | var Files = Java.type(" | ||
+ | var BinaryValueObject = Java.type(" | ||
+ | var LongValueObject = Java.type(" | ||
+ | var arrayOfBytes = Java.type(" | ||
+ | |||
+ | recipe = Files.readAllBytes(java.nio.file.Paths.get(" | ||
+ | // Compress the bytes | ||
+ | output = new arrayOfBytes(2000); | ||
+ | compresser = new Deflater(); | ||
+ | compresser.setInput(recipe); | ||
+ | compresser.finish(); | ||
+ | compressedDataLength = compresser.deflate(output); | ||
+ | compresser.end(); | ||
+ | outArray = new arrayOfBytes(compressedDataLength); | ||
+ | java.lang.System.arraycopy(output, | ||
+ | |||
+ | // save the length of the compress recipe to the PLC using a server called " | ||
+ | CompressedRecipeLength-> | ||
+ | |||
+ | new BinaryValueObject(outArray); | ||
+ | </ | ||
+ | |||
+ | The inverse of the above code reads the bytes from the PLC and stores them to a file. | ||
+ | |||
+ | < | ||
+ | var Deflater = Java.type(" | ||
+ | var Inflater = Java.type(" | ||
+ | var Files = Java.type(" | ||
+ | var BinaryValueObject = Java.type(" | ||
+ | var StringValueObject = Java.type(" | ||
+ | var arrayOfBytes = Java.type(" | ||
+ | |||
+ | // read the length of the compress recipe from the PLC using a server called " | ||
+ | CompressedRecipeLength-> | ||
+ | compressedDataLength = CompressedRecipeLength-> | ||
+ | |||
+ | java.lang.System.out.println | ||
+ | // Compress the bytes | ||
+ | output = new arrayOfBytes(2000); | ||
+ | inflater = new Inflater(); | ||
+ | buf = incomingValue.getBinaryValue(); | ||
+ | |||
+ | inflater.setInput(buf, | ||
+ | |||
+ | resultLength = inflater.inflate(output); | ||
+ | inflater.end(); | ||
+ | |||
+ | // trim the array to the size of of the output | ||
+ | outArray = new arrayOfBytes(resultLength); | ||
+ | java.lang.System.arraycopy(output, | ||
+ | |||
+ | Files.write(java.nio.file.Paths.get(" | ||
+ | |||
+ | </ | ||