This commit is contained in:
赵鑫 2022-09-04 15:26:20 +08:00
parent 592b94c06d
commit a961a63e1a
4 changed files with 102 additions and 285 deletions

View File

@ -1 +1 @@
# 我的树莓派 Sense Hat
# 树莓派 Sense HAT API

View File

@ -1,10 +1,10 @@
{
"name": "sensehat-server",
"version": "0.1.0",
"description": "SenseHat Server",
"version": "0.0.1",
"description": "树莓派 Sense HAT API",
"main": "server.js",
"scripts": {
"start": "nodemon",
"start": "PORT=8001 pm2 start server.js --name 'RPi Sense HAT API:8001' --watch",
"test": "echo \"Error: no test specified\" && exit 1"
},
"repository": {
@ -23,8 +23,5 @@
"request": "^2.88.2",
"sense-hat-led": "^1.2.0",
"socket.io": "^4.5.1"
},
"devDependencies": {
"nodemon": "^2.0.19"
}
}

View File

@ -1,170 +0,0 @@
<!DOCTYPE html>
<html lang="zh">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="shortcut icon" href="data:" type="image/x-icon">
<title>Sense Hat</title>
<style>
* {
box-sizing: border-box;
}
#matrix {
display: flex;
flex-direction: column;
gap: 2px;
}
.row {
display: flex;
width: 94px;
gap: 2px;
}
.led {
width: 10px;
height: 10px;
border: 1px solid gray;
background-color: whitesmoke;
cursor: pointer;
border-radius: 2px;
}
.led.on {
background-color: blue;
}
button {
width: 94px;
margin-top: 1rem;
}
</style>
</head>
<body>
<div id="matrix">
<div class="row">
<div class="led" data-y="0" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="0" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="1" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="1" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="2" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="2" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="3" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="3" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="4" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="4" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="5" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="5" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="6" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="6" data-x="7" onclick="turnon(this)"></div>
</div>
<div class="row">
<div class="led" data-y="7" data-x="0" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="1" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="2" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="3" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="4" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="5" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="6" onclick="turnon(this)"></div>
<div class="led" data-y="7" data-x="7" onclick="turnon(this)"></div>
</div>
</div>
<button onclick="client.emit('action', {action:'clear'})">clear all</button>
<br>
<button onclick="client.emit('action', {action:'hello'})">hello</button>
<br>
<button onclick="client.emit('action', {action:'flash'})">flash</button>
<br>
<button onclick="client.emit('action', {action:'temp'})">temp</button>
<div id="rpis"></div>
<script src="socket.io/socket.io.js"></script>
<script>
const client = io()
function turnon(led) {
const { x, y } = led.dataset
const action = led.classList.toggle('on') ? 'on' : 'off'
client.emit('action', { action, x, y })
}
client.on('leds', (array) => {
const leds = document.querySelectorAll('.led')
for (let y = 0; y < 8; y++) {
for (let x = 0; x < 8; x++) {
const n = y * 8 + x
if (array[y][x] == 1) leds[n].classList.add('on')
else leds[n].classList.remove('on')
}
}
})
client.on('data', (data) => {
rpis.innerHTML = ''
data.forEach(rpi => {
const p = document.createElement('p')
p.innerHTML = `hostname: ${rpi.hostname}<br>cpu load: ${rpi.cpu.load}<br>cpu temperature: ${rpi.cpu.temperature}<br>memory total: ${rpi.memory.total}<br>memory available: ${rpi.memory.available}`
rpis.appendChild(p)
})
})
</script>
</body>
</html>

208
server.js
View File

@ -1,70 +1,60 @@
require('dotenv').config()
const port = process.env.PORT || 3000
const host = process.env.HOST || 'localhost'
const express = require('express')
const socketio = require('socket.io')
const CronJob = require('cron').CronJob
const request = require('request')
// const CronJob = require('cron').CronJob
// const request = require('request')
const sense = require('sense-hat-led').sync
const app = express()
const server = app.listen(port, '0.0.0.0', () => console.log(`server is running on port ${port}`))
const server = app.listen(port, host, () => console.log(`rpi sense hat server is running at port http://${host}:${port}`))
const io = socketio(server)
app.use(express.static('public'))
sense.clear()
sense.setRotation(180)
// Sense HAT 初始化
sense.lowLight = true
sense.setRotation(180)
sense.clear()
const color_red = [255, 0, 0]
const color_blue = [0, 0, 255]
const leds = [
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
[0, 0, 0, 0, 0, 0, 0, 0],
]
io.emit('leds', leds)
let zhao_server_errors = 0
const job = new CronJob(
'*/3 * * * * *',
async () => {
// 测试 http://zhao 的连通性
request('http://zhao', (err, res, body) => {
if (res.statusCode != 200) {
zhao_server_errors++
}
if (zhao_server_errors > 0) {
for (let i = 0; i < zhao_server_errors; i++) {
const x = i % 8
const y = Math.floor(i / 8)
sense.setPixel(x, y, color_red)
}
} else {
let color = sense.getPixel(0, 0)
let [r, g, b] = color
sense.setPixel(0, 0, [r, g > 0 ? 0 : 255, b])
}
io.on('connection', (client) => {
console.log(`client conncted: ${client.id}`)
client.emit('message', `hello, ${client.id}`)
})
},
null,
true
)
// const rpistatus = new CronJob(
// '*/5 * * * * *',
// const color_red = [255, 0, 0]
// const color_blue = [0, 0, 255]
// const leds = [
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// [0, 0, 0, 0, 0, 0, 0, 0],
// ]
// io.emit('leds', leds)
// let zhao_server_errors = 0
// const job = new CronJob(
// '*/3 * * * * *',
// async () => {
// request('http://zhao:4000', (err, res, body) => {
// if (res.statusCode == 200) {
// io.emit('data', body)
// // 测试 http://zhao 的连通性
// request('http://zhao', (err, res, body) => {
// if (res.statusCode != 200) {
// zhao_server_errors++
// }
// })
// request('http://pi:4000', (err, res, body) => {
// if (res.statusCode == 200) {
// io.emit('data', body)
// if (zhao_server_errors > 0) {
// for (let i = 0; i < zhao_server_errors; i++) {
// const x = i % 8
// const y = Math.floor(i / 8)
// sense.setPixel(x, y, color_red)
// }
// } else {
// let color = sense.getPixel(0, 0)
// let [r, g, b] = color
// sense.setPixel(0, 0, [r, g > 0 ? 0 : 255, b])
// }
// })
// },
@ -72,60 +62,60 @@ const job = new CronJob(
// true
// )
io.on('connection', (client) => {
client.emit('leds', leds)
client.on('action', ({ action, x, y }) => {
console.log({ action, x, y })
switch (action) {
case 'on':
sense.setPixel(Number(x), Number(y), color_blue)
leds[y][x] = 1
break
case 'off':
sense.setPixel(Number(x), Number(y), [0, 0, 0])
leds[y][x] = 0
break
case 'hello':
sense.flashMessage('HELLO', 1, color_blue)
sense.clear()
read_all_leds(0)
break
case 'flash':
sense.clear([255, 255, 255])
setTimeout(sense.clear, 100)
read_all_leds(0)
break
case 'clear':
sense.clear()
read_all_leds(0)
break
case 'temp':
read_all_leds()
get_rpis_info()
break
default:
break
}
io.emit('leds', leds)
})
})
// io.on('connection', (client) => {
// client.emit('leds', leds)
// client.on('action', ({ action, x, y }) => {
// console.log({ action, x, y })
// switch (action) {
// case 'on':
// sense.setPixel(Number(x), Number(y), color_blue)
// leds[y][x] = 1
// break
// case 'off':
// sense.setPixel(Number(x), Number(y), [0, 0, 0])
// leds[y][x] = 0
// break
// case 'hello':
// sense.flashMessage('HELLO', 1, color_blue)
// sense.clear()
// read_all_leds(0)
// break
// case 'flash':
// sense.clear([255, 255, 255])
// setTimeout(sense.clear, 100)
// read_all_leds(0)
// break
// case 'clear':
// sense.clear()
// read_all_leds(0)
// break
// case 'temp':
// read_all_leds()
// get_rpis_info()
// break
// default:
// break
// }
// io.emit('leds', leds)
// })
// })
function read_all_leds(value = null) {
const pixels = sense.getPixels()
for (let y = 0; y < 8; y++) {
for (let x = 0; x < 8; x++) {
leds[y][x] = value !== null ? value : sense.getPixel(x, y)[2] == 0 ? 0 : 1
}
}
}
// function read_all_leds(value = null) {
// const pixels = sense.getPixels()
// for (let y = 0; y < 8; y++) {
// for (let x = 0; x < 8; x++) {
// leds[y][x] = value !== null ? value : sense.getPixel(x, y)[2] == 0 ? 0 : 1
// }
// }
// }
async function get_rpis_info() {
const rpis = []
request('http://zhao:4000', (err, res, body) => {
if (res.statusCode == 200) rpis.push(JSON.parse(body))
request('http://pi:4000', (err, res, body) => {
if (res.statusCode == 200) rpis.push(JSON.parse(body))
io.emit('data', rpis)
})
})
}
// async function get_rpis_info() {
// const rpis = []
// request('http://zhao:4000', (err, res, body) => {
// if (res.statusCode == 200) rpis.push(JSON.parse(body))
// request('http://pi:4000', (err, res, body) => {
// if (res.statusCode == 200) rpis.push(JSON.parse(body))
// io.emit('data', rpis)
// })
// })
// }