local wakeup = require("+wakeup") --- Builds and returns a Wake-on-LAN magic packet for [[mac]] address. local function magic_packet (mac) local data = "" for octet in mac:gmatch("%x%x") do data = data .. string.char(tonumber(octet, 16)) end if #data ~= 6 then error(string.format("Invalid MAC\t%q", mac), 2) end return string.char(0xff):rep(6) .. data:rep(16) end function wakeup:open () self.udp = net.createUDPSocket() end function wakeup:close () if self.udp then self.udp:close() end end --- Creates a new counter that has :up() method that returns true until it gets called N [[times]]. Then it returns only --- false. local function new_counter (times) return { count = 0, times = times, up = function (self) self.count = self.count + 1 return self.count <= self.times end, } end --- Begins to send Wake-on-LAN magic packets to the broadcast [[.addr]] for selected [[.mac]]. Packets are sent several --- [[times]] each separated by [[interval]]. Callback is called [[after]] all packets are sent. function wakeup:sendout (interval, times, after) self:open() local count = new_counter(times) return tmr.create():alarm(interval, tmr.ALARM_SEMI, function (timer) if count:up() then print("Waking up", self.mac, self.addr) self.udp:send(self.port, self.addr, self.data) timer:start() else timer:unregister() self:close() after() end end) end function wakeup:now () local time = rtctime.epoch2cal(rtctime.get()) time.year = nil time.sec = nil return time end local function compare (time, pattern) for key, value in pairs(pattern) do if time[key] ~= value then return false end end return true end function wakeup:init () gpio.write(4, gpio.HIGH) gpio.mode(4, gpio.OUTPUT) self.data = magic_packet(self.mac) wifi.eventmon.register(wifi.eventmon.STA_GOT_IP, function () sntp.sync( self.sntp, function () if compare(self:now(), self.when) then gpio.write(4, gpio.LOW) self:sendout(500, 5, function () wifi.setmode(wifi.NULLMODE, false) gpio.write(4, gpio.HIGH) end) else print("Not yet time") wifi.setmode(wifi.NULLMODE, false) tmr.create():alarm(25000, tmr.ALARM_SINGLE, function () self:run() end) end end, function (code, err) print("SNTP sync failed", code, err) node.restart() end) end) end function wakeup:run () wifi.setmode(wifi.STATION, false) wifi.sta.connect() end return wakeup