pimoroni-pico/micropython/examples/common/network_manager.py

109 lines
3.5 KiB
Python
Raw Normal View History

2022-06-30 10:20:18 +01:00
import rp2
import network
import machine
import uasyncio
class NetworkManager:
_ifname = ("Client", "Access Point")
def __init__(self, country="GB", client_timeout=60, access_point_timeout=5, status_handler=None, error_handler=None):
2022-06-30 10:20:18 +01:00
rp2.country(country)
self._ap_if = network.WLAN(network.AP_IF)
self._sta_if = network.WLAN(network.STA_IF)
self._mode = network.STA_IF
self._client_timeout = client_timeout
self._access_point_timeout = access_point_timeout
self._status_handler = status_handler
2022-07-14 16:53:32 +01:00
self._error_handler = error_handler
2022-06-30 10:20:18 +01:00
self.UID = ("{:02X}" * 8).format(*machine.unique_id())
def isconnected(self):
return self._sta_if.isconnected() or self._ap_if.isconnected()
2022-07-14 16:53:32 +01:00
def config(self, var):
if self._sta_if.active():
return self._sta_if.config(var)
else:
if var == "password":
return self.UID
return self._ap_if.config(var)
def mode(self):
if self._sta_if.isconnected():
return self._ifname[0]
if self._ap_if.isconnected():
return self._ifname[1]
return None
2022-06-30 10:20:18 +01:00
def ifaddress(self):
if self._sta_if.isconnected():
return self._sta_if.ifconfig()[0]
if self._ap_if.isconnected():
return self._ap_if.ifconfig()[0]
return '0.0.0.0'
def disconnect(self):
if self._sta_if.isconnected():
self._sta_if.disconnect()
if self._ap_if.isconnected():
self._ap_if.disconnect()
async def wait(self, mode):
while not self.isconnected():
self._handle_status(mode, None)
await uasyncio.sleep_ms(1000)
def _handle_status(self, mode, status):
2022-07-14 16:53:32 +01:00
if callable(self._status_handler):
2022-06-30 10:20:18 +01:00
self._status_handler(self._ifname[mode], status, self.ifaddress())
2022-07-14 16:53:32 +01:00
def _handle_error(self, mode, msg):
if callable(self._error_handler):
if self._error_handler(self._ifname[mode], msg):
return
raise RuntimeError(msg)
2022-06-30 10:20:18 +01:00
async def client(self, ssid, psk):
if self._sta_if.isconnected():
2022-07-14 16:53:32 +01:00
self._handle_status(network.STA_IF, True)
2022-06-30 10:20:18 +01:00
return
self._ap_if.disconnect()
self._ap_if.active(False)
self._sta_if.active(True)
self._sta_if.config(pm=0xa11140)
2022-06-30 10:20:18 +01:00
self._sta_if.connect(ssid, psk)
try:
await uasyncio.wait_for(self.wait(network.STA_IF), self._client_timeout)
self._handle_status(network.STA_IF, True)
except uasyncio.TimeoutError:
self._sta_if.active(False)
self._handle_status(network.STA_IF, False)
2022-07-14 16:53:32 +01:00
self._handle_error(network.STA_IF, "WIFI Client Failed")
2022-06-30 10:20:18 +01:00
async def access_point(self):
if self._ap_if.isconnected():
2022-07-14 16:53:32 +01:00
self._handle_status(network.AP_IF, True)
2022-06-30 10:20:18 +01:00
return
self._sta_if.disconnect()
self._sta_if.active(False)
2022-07-14 16:53:32 +01:00
self._ap_if.ifconfig(("10.10.1.1", "255.255.255.0", "10.10.1.1", "10.10.1.1"))
self._ap_if.config(password=self.UID)
2022-06-30 10:20:18 +01:00
self._ap_if.active(True)
try:
await uasyncio.wait_for(self.wait(network.AP_IF), self._access_point_timeout)
self._handle_status(network.AP_IF, True)
except uasyncio.TimeoutError:
self._sta_if.active(False)
self._handle_status(network.AP_IF, False)
2022-07-14 16:53:32 +01:00
self._handle_error(network.AP_IF, "WIFI Client Failed")