main.lua 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. -- LuaTools需要PROJECT和VERSION这两个信息
  2. PROJECT = "sntpdemo"
  3. VERSION = "1.0.0"
  4. --[[
  5. 本demo需要mqtt库, 大部分能联网的设备都具有这个库
  6. mqtt也是内置库, 无需require
  7. ]]
  8. -- sys库是标配
  9. _G.sys = require("sys")
  10. --[[特别注意, 使用http库需要下列语句]]
  11. _G.sysplus = require("sysplus")
  12. -- Air780E的AT固件默认会为开机键防抖, 导致部分用户刷机很麻烦
  13. if rtos.bsp() == "EC618" and pm and pm.PWK_MODE then
  14. pm.power(pm.PWK_MODE, false)
  15. end
  16. -- 统一联网函数
  17. sys.taskInit(function()
  18. local device_id = mcu.unique_id():toHex()
  19. -----------------------------
  20. -- 统一联网函数, 可自行删减
  21. ----------------------------
  22. if wlan and wlan.connect then
  23. -- wifi 联网, ESP32系列均支持
  24. local ssid = "luatos1234"
  25. local password = "12341234"
  26. log.info("wifi", ssid, password)
  27. -- TODO 改成自动配网
  28. -- LED = gpio.setup(12, 0, gpio.PULLUP)
  29. wlan.init()
  30. wlan.setMode(wlan.STATION) -- 默认也是这个模式,不调用也可以
  31. device_id = wlan.getMac()
  32. wlan.connect(ssid, password, 1)
  33. elseif mobile then
  34. -- Air780E/Air600E系列
  35. --mobile.simid(2) -- 自动切换SIM卡
  36. -- LED = gpio.setup(27, 0, gpio.PULLUP)
  37. device_id = mobile.imei()
  38. elseif w5500 then
  39. -- w5500 以太网, 当前仅Air105支持
  40. w5500.init(spi.HSPI_0, 24000000, pin.PC14, pin.PC01, pin.PC00)
  41. w5500.config() --默认是DHCP模式
  42. w5500.bind(socket.ETH0)
  43. -- LED = gpio.setup(62, 0, gpio.PULLUP)
  44. elseif socket then
  45. -- 适配的socket库也OK
  46. -- 没有其他操作, 单纯给个注释说明
  47. else
  48. -- 其他不认识的bsp, 循环提示一下吧
  49. while 1 do
  50. sys.wait(1000)
  51. log.info("bsp", "本bsp可能未适配网络层, 请查证")
  52. end
  53. end
  54. -- 默认都等到联网成功
  55. sys.waitUntil("IP_READY")
  56. sys.publish("net_ready", device_id)
  57. end)
  58. sys.taskInit(function()
  59. -- 等待联网
  60. local ret, device_id = sys.waitUntil("net_ready")
  61. sys.wait(1000)
  62. -- 对于Cat.1模块, 移动/电信卡, 通常会下发基站时间, 那么sntp就不是必要的, 而联通卡通常不会下发, 就需要sntp了
  63. -- 对应ESP32系列模块, 固件默认也会执行sntp, 所以手动调用sntp也是可选的
  64. -- sntp内置了几个常用的ntp服务器, 也支持自选服务器
  65. while 1 do
  66. -- 使用内置的ntp服务器地址, 包括阿里ntp
  67. log.info("开始执行SNTP")
  68. socket.sntp()
  69. -- 自定义ntp地址
  70. -- socket.sntp("ntp.aliyun.com")
  71. -- socket.sntp({"baidu.com", "abc.com", "ntp.air32.cn"})
  72. -- 通常只需要几百毫秒就能成功
  73. local ret = sys.waitUntil("NTP_UPDATE", 5000)
  74. if ret then
  75. -- 以下是获取/打印时间的演示,注意时区问题
  76. log.info("sntp", "时间同步成功", "本地时间", os.date())
  77. log.info("sntp", "时间同步成功", "UTC时间", os.date("!%c"))
  78. log.info("sntp", "时间同步成功", "RTC时钟(UTC时间)", json.encode(rtc.get()))
  79. -- os.time(rtc.get()) 需要 2023.07.21 之后的版本, 因为月份的命名差异mon/month
  80. -- log.info("sntp", "时间同步成功", "utc时间戳", os.time(rtc.get()))
  81. log.info("sntp", "时间同步成功", "本地时间戳", os.time())
  82. local t = os.date("*t")
  83. log.info("sntp", "时间同步成功", "本地时间os.date() json格式", json.encode(t))
  84. log.info("sntp", "时间同步成功", "本地时间os.date(os.time())", os.time(t))
  85. -- log.info("sntp", "时间同步成功", "本地时间", os.time())
  86. -- 正常使用, 一小时一次, 已经足够了, 甚至1天一次也可以
  87. -- sys.wait(3600000)
  88. -- 这里为了演示, 用5秒一次
  89. sys.wait(5000)
  90. else
  91. log.info("sntp", "时间同步失败")
  92. sys.wait(60000) -- 1分钟后重试
  93. end
  94. -- 时间戳, 精确到毫秒. 2023.11.15 新增
  95. -- 注意, 至少成功完成2次sntp,该时间戳才比较准确
  96. -- 如果仅完成了一次sntp, 时间戳比标准时间会慢一个网络延时的时长(10~500ms)不等
  97. if socket.ntptm then
  98. local tm = socket.ntptm()
  99. log.info("tm数据", json.encode(tm))
  100. log.info("时间戳", string.format("%u.%03d", tm.tsec, tm.tms))
  101. sys.wait(5000)
  102. end
  103. end
  104. end)
  105. sys.subscribe("NTP_ERROR", function()
  106. log.info("socket", "sntp error")
  107. -- socket.sntp()
  108. end)
  109. -- 用户代码已结束---------------------------------------------
  110. -- 结尾总是这一句
  111. sys.run()
  112. -- sys.run()之后后面不要加任何语句!!!!!