Simple Client Application For SR-201
Simple Client Application For SR-201
https://github.com/berkinet/sr-201-relay/blob/master/sr-201-relay.py
Resently I ordered a little board 50mm x 70mm with two relays. They can be switched
by sending commands over TCP or UDP. The only problem with it is, that the code
examples and instruction manual are entierly written in Chinese. Therefore, I created
this repo to keep track of my findings regarding the SR-201-2.
Models
The same idea, switching relays over ethernet, resulted in at least four different models
of the SR-201:
They all seem to work with the same chip and software. Although, e.g., the SR-201-2
only has two relays, it also has an extension port with another 6 pins which can be
switched, too.
When connected over TCP (port 6722), the board will only accept 6 connections at a
time. To prevent starving, it will close TCP connection after they have been idle for 15
seconds.
Since UDP (port 6723) is not an end-to-end connection, there are no restrictions. But it
is noteworthy that the board will execute UDP commands, but it will never answer.
Therefore querying the state of the relays has to be done over TCP.
The board also listens to the TCP port 5111. Over this connection the board can be
configured. E.g., its static IP address can be changed.
Factory Defaults
Static IP address : 192.168.1.100
Subnet mask : 255.255.255.0
Default Gateway : 192.168.1.1
Persistent relay state when power is lost : off
Cloud service password : 000000
DNS Server : 192.168.1.1
Cloud service : connect.tutuuu.com
Cloud service enabled: false
Example Code
This repo contains the following modules:
Maven will create an executable JAR in each of the modules target directories.
Scripts
In addition to my Java code examples that are clearly intended as a replacement for the
default VB and Delphi programs, I added a scripts directory that contains simpler more
pragmatic approaches to the SR-201 communication scheme.
Own Scripts
If you want to quickly setup your SR-201 without even starting a script or anything else,
just check the protocol Config commands and e.g. send a command via netcat:
Note: It is crucial to use printf here, as newlines are seen as errors. It drove me crazy to
find out about this one.
Python
#!/usr/bin/python
# SYNOPSIS
# DESCRIPTION
# BUGS
# Although this program can be used to issue control commands to the relay
# reliable, and the task is so simple the code will be shorter. The
# primary purpose of this code is to document how to do it, with a working
# example.
# The device can be reset to these defaults by shorting the CLR pins
# on the header next to the RJ45 connector. CLR is adjacent to the +5V
# THE DEVICE: Commands that can be sent over the TCP and UDP control ports
#
# 0R No operation (but return status).
# 1R* Close relay if it's open, wait approx 1/2 a second, open
# relay.
# 1R:n Close relay if it's open, then in n seconds (1 <= n <= 65535)
# open it.
# Where:
# R is the relay number, '1' .. '8'. The main board has relay's
# '1' and '2', the extension board (if present) has the rest.
#
# If the command is sent over TCP (not UDP, TCP only), the relay will
# reply with a string of 8 0's and 1's, representing the "before" command
# was executed" state of relay's 1..8 in that order. A '0' is sent if the
# Commands are ASCII strings that must be sent in one TCP packet. 'i'
#
>192.168.1.100,255.255.255.0,192.168.1.1,,0,435,F44900F6087457000000,192.168.1.
1,connect.tutuuu.com,0;
# 4 192.168.1.1 Gateway.
# 5 Unknown.
# #Di,F; Set the state whose ID is 'D' to value 'F'. For example:
#
#
# If cloud operation is enabled (by setting ID "A" to "1"), the device sends
# User-Agent: SR-201W/M96Y
# Content-Type: application/json
# Host: 192.168.1.1
# Content-Length: 30
# "F0123456789ABCXXXXXX00000000"
# F0123456789ABCXXXXXX00000000
# \------------/\----/\------/
#|||
# | | +---- State of relays 1..8, 0=open, 1=closed.
# | +----------- Password.
# The response must have the Content-Length header set, and the body must
# HTTP/1.1 200 OK
# Content-Type: application/json
# Content-Length: 7
# "A11:2"
# License
# -------
# it under the terms of the GNU Affero General Public License as published
# of the GNU Affero General Public License, version 3, exempting you from
# Program or any work based on the Program. You are still required to
# Source.
# You should have received a copy of the GNU Affero General Public License
import os
import re
import select
import socket
import sys
import time
class Sr201(object):
_IPV4_RE = '[.]'.join(
('(?:[0-9]{1,2}|[01][0-9]{2}|2[0-4][0-9]|25[0-5])',) * 4)
_hostname = None
_port = None
_soc = None
trace = False
CONFIG_NAMES = [
'ip',
'netmask',
'gateway',
'(blank)',
'power_persist',
'version',
'serial',
'dns',
'cloud_server',
'cloud_enabled',
'cloud_password']
PORT_CONFIG = 5111
PORT_CONTROL = 6722
MAX_RELAYS = 2
self.open()
# communications functions
def flush(self):
while s[0]:
data = self._soc.recv(4096)
if self.trace:
if port == self._port:
self.flush()
return
self.close()
self._soc = socket.create_connection((self._hostname, port))
self._port = port
def close(self):
if self._soc:
self.flush()
self._soc.close()
self._port = None
self._soc = None
def recv(self):
response = self._soc.recv(4096).decode('latin1')
if self.trace:
return response
if self.trace:
return self._soc.send(data.encode('latin1'))
#
# operations functions
if not match:
self.open()
self.send(data)
states = self.recv()
relayNum = int(match.group(1))
relayState = states[relayNum - 1]
sys.stdout.write(relayState)
match = 'close:([1-8Xx])([~]|:[1-9][0-9]{0,4}|6[0-4][0-9]{3})?$'
if not match:
usage('Invalid %r' % (command,))
if match.group(2) == '~':
data += '*'
elif match.group(2):
data += match.group(2)
self.open()
self.send(data)
states = self.recv()
relayNum = int(match.group(1))
relayState = states[relayNum - 1]
sys.stdout.write(relayState)
if command != 'status':
self.open()
self.send('00')
states = self.recv()
sys.stdout.write(states)
# config functions
self.open(self.PORT_CONFIG)
response = self.recv()
if (
):
me = os.path.basename(sys.argv[0])
sys.stderr.write(msg)
sys.exit(1)
return response
#
>192.168.1.100,255.255.255.0,192.168.1.1,,0,435,F44900F6087457000000,192.168.1.
1,connect.tutuuu.com,0;
values = response[1:-1].split(',')
if len(values) != len(self.CONFIG_NAMES) - 1:
# we will split the serial nbumber into two fields: serial number and cloud password
values.append(values[6][14:20])
values[6] = values[6][0:14]
for i in range(len(values)):
# sys.stdout.write('%s=(not-sent)\n' % (self.CONFIG_NAMES[-1],))
def do_reset(self, command):
if command != 'reset':
self.close()
if not match:
if not match:
match = (
'cloud_server=((?:[-a-z0-9]+[.])+[-a-z0-9]{2,}|' +
self._IPV4_RE + ')$')
if not match:
if not match:
if not match:
if not match:
ip = match.group(1).split('.')
if not match:
pause = float(match.group(1))
time.sleep(pause)
if pause > 10:
self.close()
if not match:
def usage(msg=None):
def split(w):
i = next(i, 64)
r = w[:i]
w[:i] = []
return ' '.join(r)
if not d:
sys.stderr.write(s + '\n')
else:
w = d.split()
while w:
me = os.path.basename(sys.argv[0])
if msg:
else:
w('options:')
w('commands:')
w(' close:R~', 'Close relay R, then in about 1/2 a second open it.')
w('R:')
sys.exit(1)
# Entry point.
def main(argv=sys.argv):
i=1
trace = False
if len(argv) > i and argv[i] == '--trace':
trace = True
i += 1
if len(argv) == i:
usage()
if argv[i].startswith("-"):
sr_201 = Sr201(argv[i])
i += 1
sr_201.trace = trace
sr_201.close()
if __name__ == '__main__':
main()