Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix scanner force-scanning #511

Merged
merged 25 commits into from
Jul 8, 2024
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
d3be0ac
Fix scanner force-scanning
uzlonewolf Jun 23, 2024
d3bc3d0
Allow a string as the force-scan list
uzlonewolf Jun 23, 2024
d088823
Add force-scanning v3.5 devices to scanner
uzlonewolf Jun 23, 2024
e9fd881
Add force-scanning v3.5 devices to scanner
uzlonewolf Jun 23, 2024
0690e8c
v1.14.2 Notes
jasonacox Jun 23, 2024
7259b8d
Rename scanner variable try_v35 to a more descriptive try_v35_with_v34
uzlonewolf Jun 24, 2024
9420ec6
Make sure set_version() is given a float (#507)
uzlonewolf Jun 24, 2024
133fd35
Allow host bits in the force-scan network list
uzlonewolf Jun 25, 2024
da6d01b
Refactor _print_device_info function to include verbose flag
jasonacox Jun 25, 2024
51f7653
Merge branch 'forcescan-fix' of https://github.com/uzlonewolf/tinytuy…
jasonacox Jun 25, 2024
7d160a0
Allow device discovery packets on port 7000
uzlonewolf Jul 7, 2024
ff6abf2
Add get_ip_to_broadcast() function to scanner
uzlonewolf Jul 7, 2024
5f38b49
Add "Force Scan" button and UI updates
jasonacox Jul 7, 2024
59a013f
Merge branch 'forcescan-fix' of https://github.com/uzlonewolf/tinytuy…
jasonacox Jul 7, 2024
831f113
Make scanner send broadcasts to port 7000
uzlonewolf Jul 7, 2024
4d46538
Update control panel image
jasonacox Jul 7, 2024
6073a8c
Add port 7000 support to tools/pcap_parse.py
uzlonewolf Jul 7, 2024
3610184
Merge branch 'forcescan-fix' of github-lw:uzlonewolf/tinytuya into fo…
uzlonewolf Jul 7, 2024
a20b080
Fix incorrect message log level
uzlonewolf Jul 7, 2024
2fbb691
Allow calling send_discovery_request() without an argument
uzlonewolf Jul 7, 2024
0b19990
Rewrite tinytuya.find_device() to use the scanner
uzlonewolf Jul 7, 2024
c6dba30
Also use psutil to get the force-scan IP list if netifaces is not ava…
uzlonewolf Jul 7, 2024
b00882a
Close the broadcast sockets once we're done with them
uzlonewolf Jul 7, 2024
728779d
Add broadcast for 3.5 devices to server
jasonacox Jul 7, 2024
3aea261
Release notes
jasonacox Jul 8, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions RELEASE.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,9 @@
# RELEASE NOTES

## v1.14.1 - Scanner Fixes

* Fix force-scanning bug in scanner introduced in last release by @uzlonewolf in https://github.com/jasonacox/tinytuya/pull/511.

## v1.14.0 - Command Line Updates

* PyPI 1.14.0 rewrite of main to use argparse and add additional options by @uzlonewolf in https://github.com/jasonacox/tinytuya/pull/503
Expand Down
8 changes: 7 additions & 1 deletion server/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -104,12 +104,18 @@ docker start tinytuya

The UI at http://localhost:8888 allows you to view and control the devices.

![image](https://user-images.githubusercontent.com/836718/227736045-adb6e359-c0c1-44b9-b9ad-7e978f6b7b84.png)
![image](https://github.com/jasonacox/tinytuya/assets/836718/e00a1f9a-48e2-400c-afa1-7a81799efa89)

![image](https://user-images.githubusercontent.com/836718/227736057-e5392c13-554f-457e-9082-43c4d41a98ed.png)

## Release Notes

### t12 - Force Scan

* Added "Force Scan" button to cause server to run a network scan for devices not broadcasting.
* Minor updates to UI for a cleaner title and footer to accommodate button.
* Added logic to allow settings via environmental variables.

### t11 - Minimize Container

* Reduce size of Docker container by removing rust build and using python:3.12-bookworm.
Expand Down
106 changes: 77 additions & 29 deletions server/server.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@
import sys
import os
import urllib.parse
from http.server import BaseHTTPRequestHandler, HTTPServer, ThreadingHTTPServer
from http.server import BaseHTTPRequestHandler, HTTPServer
from socketserver import ThreadingMixIn

# Terminal color capability for all platforms
Expand All @@ -59,30 +59,28 @@
pass

import tinytuya
from tinytuya import scanner
import os

BUILD = "t11"

# Defaults
APIPORT = 8888
DEBUGMODE = False
DEVICEFILE = tinytuya.DEVICEFILE
SNAPSHOTFILE = tinytuya.SNAPSHOTFILE
CONFIGFILE = tinytuya.CONFIGFILE
TCPTIMEOUT = tinytuya.TCPTIMEOUT # Seconds to wait for socket open for scanning
TCPPORT = tinytuya.TCPPORT # Tuya TCP Local Port
MAXCOUNT = tinytuya.MAXCOUNT # How many tries before stopping
UDPPORT = tinytuya.UDPPORT # Tuya 3.1 UDP Port
UDPPORTS = tinytuya.UDPPORTS # Tuya 3.3 encrypted UDP Port
UDPPORTAPP = tinytuya.UDPPORTAPP # Tuya App
TIMEOUT = tinytuya.TIMEOUT # Socket Timeout
RETRYTIME = 30
RETRYCOUNT = 5
SAVEDEVICEFILE = True

# Check for Environmental Overrides
debugmode = os.getenv("DEBUG", "no")
if debugmode.lower() == "yes":
DEBUGMODE = True
BUILD = "t12"

# Defaults from Environment
APIPORT = int(os.getenv("APIPORT", "8888"))
DEBUGMODE = os.getenv("DEBUGMODE", "False").lower() == "true"
DEVICEFILE = os.getenv("DEVICEFILE", tinytuya.DEVICEFILE)
SNAPSHOTFILE = os.getenv("SNAPSHOTFILE", tinytuya.SNAPSHOTFILE)
CONFIGFILE = os.getenv("CONFIGFILE", tinytuya.CONFIGFILE)
TCPTIMEOUT = float(os.getenv("TCPTIMEOUT", str(tinytuya.TCPTIMEOUT)))
TCPPORT = int(os.getenv("TCPPORT", str(tinytuya.TCPPORT)))
MAXCOUNT = int(os.getenv("MAXCOUNT", str(tinytuya.MAXCOUNT)))
UDPPORT = int(os.getenv("UDPPORT", str(tinytuya.UDPPORT)))
UDPPORTS = int(os.getenv("UDPPORTS", str(tinytuya.UDPPORTS)))
UDPPORTAPP = int(os.getenv("UDPPORTAPP", str(tinytuya.UDPPORTAPP)))
TIMEOUT = float(os.getenv("TIMEOUT", str(tinytuya.TIMEOUT)))
RETRYTIME = int(os.getenv("RETRYTIME", "30"))
RETRYCOUNT = int(os.getenv("RETRYCOUNT", "5"))
SAVEDEVICEFILE = os.getenv("SAVEDEVICEFILE", "True").lower() == "true"
DEBUGMODE = os.getenv("DEBUGMODE", "no").lower() == "yes"

# Logging
log = logging.getLogger(__name__)
Expand Down Expand Up @@ -124,7 +122,8 @@ def sig_term_handle(signum, frame):
retrydevices = {}
retrytimer = 0
cloudconfig = {'apiKey':'', 'apiSecret':'', 'apiRegion':'', 'apiDeviceID':''}

forcescan = False
forcescandone = True

# Terminal formatting
(bold, subbold, normal, dim, alert, alertdim, cyan, red, yellow) = tinytuya.termcolor(True)
Expand Down Expand Up @@ -310,6 +309,7 @@ def tuyalisten(port):
result["mac"] = mac
result["key"] = dkey
result["id"] = gwId
result["forced"] = False

# add device if new
if not appenddevice(result, deviceslist):
Expand All @@ -323,7 +323,6 @@ def tuyalisten(port):

class ThreadingHTTPServer(ThreadingMixIn, HTTPServer):
daemon_threads = True
pass

def delayoff(d, sw):
d.turn_off(switch=sw, nowait=True)
Expand Down Expand Up @@ -355,8 +354,10 @@ def do_POST(self):
self.wfile.write(bytes(message, "utf8"))

def do_GET(self):
# pylint: disable=global-variable-not-assigned
global retrytimer, retrydevices
global cloudconfig, deviceslist
global forcescan, forcescandone

self.send_response(200)
message = "Error"
Expand Down Expand Up @@ -516,6 +517,8 @@ def do_GET(self):
jout = {}
jout["found"] = len(deviceslist)
jout["registered"] = len(tuyadevices)
jout["forcescan"] = forcescan
jout["forcescandone"] = forcescandone
message = json.dumps(jout)
elif self.path.startswith('/status/'):
id = self.path.split('/status/')[1]
Expand Down Expand Up @@ -561,6 +564,11 @@ def do_GET(self):
retrydevices['*'] = 1
elif self.path == '/offline':
message = json.dumps(offlineDevices())
elif self.path == '/scan':
# Force Scan for new devices
forcescan = True
forcescandone = False
message = json.dumps({"OK": "Forcing a scan for new devices."})
else:
# Serve static assets from web root first, if found.
fcontent, ftype = get_static(web_root, self.path)
Expand Down Expand Up @@ -606,7 +614,7 @@ def api(port):
tuyaUDPs = threading.Thread(target=tuyalisten, args=(UDPPORTS,))
tuyaUDP7 = threading.Thread(target=tuyalisten, args=(UDPPORTAPP,))
apiServer = threading.Thread(target=api, args=(APIPORT,))

print(
"\n%sTinyTuya %s(Server)%s [%s%s]\n"
% (bold, normal, dim, tinytuya.__version__, BUILD)
Expand All @@ -626,10 +634,50 @@ def api(port):

print(" * API and UI Endpoint on http://localhost:%d" % APIPORT)
log.debug("Server URL http://localhost:%d" % APIPORT)

try:
while(True):
log.debug("Discovered Devices: %d " % len(deviceslist))
if forcescan:
print(" + ForceScan: Scan for new devices started...")
forcescan = False
retrytimer = time.time() + RETRYTIME
# def devices(verbose=False, scantime=None, color=True, poll=True, forcescan=False, byID=False, show_timer=None,
# discover=True, wantips=None, wantids=None, snapshot=None, assume_yes=False, tuyadevices=[],
# maxdevices=0)
try:
found = scanner.devices(forcescan=True, verbose=False, discover=False, tuyadevices=tuyadevices)
except:
log.error("Error during scanner.devices()")
found = []
print(f" - ForceScan: Found {len(found)} devices")
for f in found:
log.debug(f" - {found[f]}")
gwId = found[f]["id"]
result = {}
dname = dkey = mac = ""
try:
# Try to pull name and key data
(dname, dkey, mac) = tuyaLookup(gwId)
except:
pass
# set values
result["name"] = dname
result["mac"] = mac
result["key"] = dkey
result["id"] = gwId
result["ip"] = found[f]["ip"]
result["version"] = found[f]["version"]
result["forced"] = True

# add device if new
if not appenddevice(result, deviceslist):
# Added device to list
if dname == "" and dkey == "" and result["id"] not in newdevices:
# If fetching the key failed, save it to retry later
retrydevices[result["id"]] = RETRYCOUNT
newdevices.append(result["id"])
forcescandone = True

if retrytimer <= time.time() or '*' in retrydevices:
if len(retrydevices) > 0:
Expand Down Expand Up @@ -671,4 +719,4 @@ def api(port):
# Close down API thread
print("Stopping threads...")
log.debug("Stoppping threads")
requests.get('http://localhost:%d/stop' % APIPORT)
requests.get('http://localhost:%d/stop' % APIPORT, timeout=5)
39 changes: 35 additions & 4 deletions server/web/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
<html lang="en">
<head>
<title>TinyTuya API Server</title>
<script src="https://code.jquery.com/jquery-3.2.1.min.js"></script>
<script src="jquery-3.7.1.min.js"></script>
<link rel="stylesheet" href="tinytuya.css">
</head>

Expand All @@ -18,7 +18,9 @@
<div class="offline"></div>
</div>
</div>
<div class="number"></div>
<div class="button">
<div class="number"></div>
<input id='forcescan' type='button' value='Force Scan' onclick='forcescan()'></div>
<script>

// Get Version
Expand All @@ -31,6 +33,19 @@
setTimeout(showversion, 10000);
}

// Force Scan
function forcescan() {
// Change button text and disable - restore on scan completion
// via numdevices() function
$("#forcescan").prop('value', 'Scanning...');
$("#forcescan").prop('disabled', true);
// Call the scan API
var pwurl = window.location.protocol + "//" + window.location.host + "/scan";
$.getJSON(pwurl, function(data) {
console.log(data);
});
}

// Number of Devices
function numdevices() {
var pwurl = window.location.protocol + "//" + window.location.host + "/numdevices";
Expand All @@ -39,11 +54,20 @@
` - Registered: ${data.registered}`;
var online = `Devices Online: ${data.found}`;
var numoff = data.registered - data.found;
var forcescandone = data.forcescandone;
if(numoff<0) numoff = 0;
var offline = `Devices Offline: ${numoff}`;
$(".onlinetext").html(online)
$(".offlinetext").html(offline)
$(".number").html(text);
// Update Force Scan button based on scan status
if(forcescandone) {
$("#forcescan").prop('value', 'Force Scan');
$("#forcescan").prop('disabled', false);
} else {
$("#forcescan").prop('value', 'Scanning...');
$("#forcescan").prop('disabled', true);
}
});
setTimeout(numdevices, 1000);
}
Expand All @@ -60,13 +84,15 @@
var ip = "";
var name = "";
var ver = "";
var forced = "";
for (let x in data) {
//console.log(x + ": "+ data[x])
let device = {
"name": data[x].name,
"id": x,
"ip": data[x].ip,
"version": data[x].version
"version": data[x].version,
"forced": data[x].forced
}
deviceDB.push(device)
}
Expand All @@ -77,12 +103,17 @@
output = "<table><thead>\n<tr>\n<th>#</th><th>Device Name</th><th>Device ID</th>" +
"<th>IP Address</th><th>Version</th><th>Control</th>\n</tr></thead>\n<tbody>";
for (let x in sortedDevices) {
if (sortedDevices[x].forced) {
forced = " (F)";
} else {
forced = "";
}
output = output + "<tr>\n<td>" + rownum + "</td>" +
" <td> <a href='device_dps.html?id=" + sortedDevices[x].id + "'>" +
(sortedDevices[x].name.length > 0 ? sortedDevices[x].name : '[' + sortedDevices[x].id + ']') +
"</a> </td><td>" +
"<div class='id'>" + sortedDevices[x].id + "</div></td><td>" +
"<div class='address'>" + sortedDevices[x].ip + "</div></td><td>" +
"<div class='address'>" + sortedDevices[x].ip + forced + "</div></td><td>" +
"<div class='version'>" + sortedDevices[x].version + "</div></td>" +
" <td> <a href='device_dps.html?id=" + sortedDevices[x].id + "'>View</a></td>"
"\n</tr>\n";
Expand Down
2 changes: 2 additions & 0 deletions server/web/jquery-3.7.1.min.js

Large diffs are not rendered by default.

28 changes: 28 additions & 0 deletions server/web/tinytuya.css
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,11 @@ body {
font-size: small;
}
.title {
/* Center the title at top with blue background white text */
background-color: blue;
color: white;
text-align: center;
padding: 5px;
font-size: large;
font-weight: bold;
}
Expand Down Expand Up @@ -56,6 +61,10 @@ a:hover, a:active {
padding: 5px;
}
.row {
/* Full window height minus title and button */
height: calc(100vh - 100px);
/* Allow scroll */
overflow: auto;
display: flex;
}
/* Clear floats after the columns */
Expand All @@ -67,3 +76,22 @@ a:hover, a:active {
.value {
color: rgb(73, 73, 73);
}
.button {
/* put button a bottom of the page */
position: fixed;
bottom: 0;
left: 0;
height: 40px;
width: 100%;
/* make it look like a button */
background-color: #4CAF50;
border: none;
color: white;
padding: 5px 5px;
text-align: center;
text-decoration: none;
display: inline-block;
font-size: 14px;
margin: 4px 2px;
cursor: pointer;
}
4 changes: 3 additions & 1 deletion tinytuya/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,7 @@
# Colorama terminal color capability for all platforms
init()

version_tuple = (1, 14, 0)
version_tuple = (1, 14, 1)
version = __version__ = "%d.%d.%d" % version_tuple
__author__ = "jasonacox"

Expand Down Expand Up @@ -171,6 +171,7 @@
UDP_NEW = 0x13 # 19 # FR_TYPE_ENCRYPTION
AP_CONFIG_NEW = 0x14 # 20 # FRM_AP_CFG_WF_V40
BOARDCAST_LPV34 = 0x23 # 35 # FR_TYPE_BOARDCAST_LPV34
REQ_DEVINFO = 0x25 # broadcast to port 7000 to get v3.5 devices to send their info
LAN_EXT_STREAM = 0x40 # 64 # FRM_LAN_EXT_STREAM

# Protocol Versions and Headers
Expand Down Expand Up @@ -1654,6 +1655,7 @@ def add_dps_to_request(self, dp_indicies):
self.dps_to_request.update({str(index): None for index in dp_indicies})

def set_version(self, version): # pylint: disable=W0621
version = float(version)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good call! Yes, this surfaces as an issue every 3-6mo or so.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I thought I'd sneak that one in there :) I'm not marking it as 'closes #507' as I think there's more we can do to type-safe everything.

self.version = version
self.version_str = "v" + str(version)
self.version_bytes = str(version).encode('latin1')
Expand Down
Loading
Loading