Logic and comms ready for testing

This commit is contained in:
Skylar Grant 2024-08-22 20:07:05 -04:00
parent 0c1c1483b1
commit 236faaefbe
7 changed files with 137 additions and 46 deletions

2
.vscode/launch.json vendored
View File

@ -11,7 +11,7 @@
"skipFiles": [ "skipFiles": [
"<node_internals>/**" "<node_internals>/**"
], ],
"program": "${workspaceFolder}/main.js" "program": "${workspaceFolder}/src/main.js"
} }
] ]
} }

View File

@ -5,11 +5,13 @@ module.exports = {
// State class // State class
State: class State { State: class State {
constructor(config) { constructor(config) {
this.publisher = 'backend';
this.igniter = { this.igniter = {
on: false, on: false,
name: "igniter", name: "igniter",
topic: config.mqtt.topics.igniter, topic: config.mqtt.topics.igniter,
publisher: 'backend', publisher: this.publisher,
power: (communicator, pinState) => { power: (communicator, pinState) => {
// Set the power based on the desired pinState // Set the power based on the desired pinState
this.igniter.on = pinState === 1 ? true : false; this.igniter.on = pinState === 1 ? true : false;
@ -21,7 +23,7 @@ module.exports = {
on: false, on: false,
name: "exhaust", name: "exhaust",
topic: config.mqtt.topics.exhaust, topic: config.mqtt.topics.exhaust,
publisher: 'backend', publisher: this.publisher,
power: (communicator, pinState) => { power: (communicator, pinState) => {
// Set the power based on the desired pinState // Set the power based on the desired pinState
this.exhaust.on = pinState === 1 ? true : false; this.exhaust.on = pinState === 1 ? true : false;
@ -34,13 +36,37 @@ module.exports = {
name: "auger", name: "auger",
feedRate: 500, feedRate: 500,
topic: config.mqtt.topics.auger, topic: config.mqtt.topics.auger,
publisher: 'backend', publisher: this.publisher,
power: (communicator, pinState) => { power: (communicator, pinState) => {
// Set the power based on the desired pinState // Set the power based on the desired pinState
this.auger.on = pinState === 1 ? true : false; this.auger.on = pinState === 1 ? true : false;
communicator.send(config.mqtt.topics.auger, JSON.stringify(this.auger)); communicator.send(config.mqtt.topics.auger, JSON.stringify(this.auger));
} }
}; };
this.pof = {
on: false,
name: "pof",
topic: config.mqtt.topics.pof,
publisher: this.publisher,
power: (communicator, pinState) => {
// Set the power based on the desired pinState
this.pof.on = pinState === 1 ? true : false;
communicator.send(config.mqtt.topics.pof, JSON.stringify(this.pof));
}
};
this.vacuum = {
on: false,
name: "vacuum",
topic: config.mqtt.topics.vacuum,
publisher: this.publisher,
power: (communicator, pinState) => {
// Set the power based on the desired pinState
this.vacuum.on = pinState === 1 ? true : false;
communicator.send(config.mqtt.topics.vacuum, JSON.stringify(this.vacuum));
}
};
console.log(`State initialized.`) console.log(`State initialized.`)
}; };
}, },
@ -76,19 +102,20 @@ module.exports = {
// Handle when the Broker sends us a message // Handle when the Broker sends us a message
client.on('message', (topic, message) => { client.on('message', (topic, message) => {
// Save the existing state // Save the existing state
const oldState = { ...state }; const oldState = JSON.parse(JSON.stringify(state));
// The message is a buffer which will need to be converted to string // The message is a buffer which will need to be converted to string
const msgStr = message.toString(); const msgStr = message.toString();
// Since the message is a JSON object, we can parse it // Since the message is a JSON object, we can parse it
const msgJson = JSON.parse(msgStr); const msgJson = JSON.parse(msgStr);
// Log the message // Log the message
console.log(`Message received on topic ${topic}:`); // console.log(`Message received on topic ${topic}:`);
console.log(msgJson); // console.log(msgJson);
// Check if the message is from the backend // Check if the message is from the backend
if (msgJson.publisher === this.publisher) { if (msgJson.publisher === this.publisher) {
console.log('Message is from the backend, ignoring'); // console.log('Message is from the backend, ignoring');
return; return;
} }
// console.log('Message is from the frontend, updating state');
// Update the state // Update the state
state[msgJson.name].on = msgJson.on; state[msgJson.name].on = msgJson.on;
// Emit the state change // Emit the state change

View File

@ -1,20 +1,21 @@
const { exec } = require('child_process'); const { exec } = require('child_process');
const config = require('./config.json');
module.exports = { module.exports = {
// Calls the GPIO Interface script to toggle a pin's state opposite of its current state // Calls the GPIO Interface script to toggle a pin's state opposite of its current state
togglePin(pin) { // togglePin(pin) {
return new Promise((resolve, reject) => { // return new Promise((resolve, reject) => {
exec(`python3 src/python/gpio_interface.py toggle ${pin}`, (error, stdout, stderr) => { // exec(`python3 ${config.gpioScript} toggle ${pin}`, (error, stdout, stderr) => {
if (error) reject(error); // if (error) reject(error);
if (stderr) reject(new Error(stderr)); // if (stderr) reject(new Error(stderr));
resolve(); // resolve();
}); // });
}); // });
}, // },
// Calls the GPIO Interface script to read a pin's state // Calls the GPIO Interface script to read a pin's state
readPin(pin) { readPin(pin) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
exec(`python3 src/python/gpio_interface.py read ${pin}`, (error, stdout, stderr) => { exec(`python3 ${config.gpioScript} read ${pin}`, (error, stdout, stderr) => {
if (error) reject(error); if (error) reject(error);
if (stderr) reject(new Error(stderr)); if (stderr) reject(new Error(stderr));
resolve(stdout.trim()); resolve(stdout.trim());
@ -24,7 +25,7 @@ module.exports = {
// Calls the GPIO Interface script to set a pin's state regardless of its current state // Calls the GPIO Interface script to set a pin's state regardless of its current state
setPin(pin, state) { setPin(pin, state) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
exec(`python3 src/python/gpio_interface.py set ${pin} ${state}`, (error, stdout, stderr) => { exec(`python3 ${config.gpioScript} set ${pin} ${state}`, (error, stdout, stderr) => {
if (error) { if (error) {
reject(error); reject(error);
} }

View File

@ -60,5 +60,6 @@
"stop": { "stop": {
"exhaustDelay": 600000 "exhaustDelay": 600000
} }
} },
"gpioScript": "src/python/gpio_interface.py"
} }

View File

@ -38,33 +38,11 @@ module.exports = {
}).catch(reject); }).catch(reject);
}); });
}, },
// Boot up sanity check during debug mode async init(communicator, state) {
async debugInit() {
module.exports.log('Resetting all output pins.'); module.exports.log('Resetting all output pins.');
module.exports.gpio.setDefaults().then((changes) => { module.exports.gpio.setDefaults(communicator, state).then((changes) => {
module.exports.log(changes); module.exports.log(changes);
}).catch(e => console.error(e)); }).catch(e => console.error(e));
for (const pin of pins) {
switch (pin.mode) {
case 'OUT':
gpio.togglePin(pin.board).then(() => {
module.exports.log(`Toggled ${pin.key}`);
}).catch(e => console.error(e));
// Wait 1000ms before toggling again.
await module.exports.sleep(1000);
gpio.togglePin(pin.board).then(() => {
module.exports.log(`Toggled ${pin.key}`);
}).catch(e => console.error(e));
break;
case 'IN':
gpio.readPin(pin.board).then(state => {
module.exports.log(`${pin.key} state: ${state}`);
}).catch(e => console.error(e));
default:
break;
}
};
} }
}, },
power: { power: {
@ -155,6 +133,10 @@ module.exports = {
return gpio.setPin(pin.board, 1).then(() => { return gpio.setPin(pin.board, 1).then(() => {
console.log(`${pin.key} powered on.`); console.log(`${pin.key} powered on.`);
}).catch(e => console.error(e)); }).catch(e => console.error(e));
} else {
return gpio.setPin(pin.board, 0).then(() => {
console.log(`${pin.key} powered off.`);
}).catch(e => console.error(e));
} }
} }
}); });

View File

@ -9,14 +9,19 @@ const comms = new Communicator(psState);
comms.init(psState, config); comms.init(psState, config);
fn.gpio.debugInit(); fn.gpio.init(comms, psState);
// Sensor detection loop // Sensor detection loop
setInterval(() => { setInterval(() => {
for (const pin of config.pins) { for (const pin of config.pins) {
if (pin.mode === 'IN') { if (pin.mode === 'IN') {
gpio.readPin(pin.board).then(state => { gpio.readPin(pin.board).then(state => {
const boolState = state === '1' ? true : false;
if (boolState !== psState[pin.key].on) {
psState[pin.key].on = boolState;
comms.send(config.mqtt.topics[pin.key], JSON.stringify(psState[pin.key]));
fn.log(`${pin.key}: ${state}`); fn.log(`${pin.key}: ${state}`);
}
}).catch(e => console.error(e)); }).catch(e => console.error(e));
} }
} }

75
src/python/fake_gpio.py Normal file
View File

@ -0,0 +1,75 @@
import sys
# Define the state of the pins
PIN_STATES = {
7: 0,
13: 0,
15: 0,
16: 0,
22: 0
}
# Define a function to get the pin state from the array
def get_pin_state(pin):
if pin in PIN_STATES:
return PIN_STATES[pin]
else:
print(f"Invalid pin: {pin}")
return -1
# Modify the read_pin function to use the get_pin_state function
def read_pin(pin):
try:
# Read pin state
state = get_pin_state(pin)
# Return 1 if pin is HIGH, 0 if LOW
return 1 if state == 1 else 0
except Exception as e:
# Handle errors
print(f"Error reading pin {pin}: {e}")
# Return -1 on error
return -1
# Modify the set_pin_state function to return success always
def set_pin_state(pin, state):
try:
# Set the pin state to either HIGH (1) or LOW (0)
# Exit with 0 for success
return 0
except Exception as e:
# Handle errors
print(f"Error setting pin {pin} to state {state}: {e}")
# Exit with 1 for failure
return 1
def main():
if len(sys.argv) < 3:
print("Usage: python3 gpio_interface.py <command> <pin> [<state>]")
sys.exit(1)
command = sys.argv[1].lower()
pin = int(sys.argv[2])
if command == "toggle":
result = toggle_pin(pin)
sys.exit(result)
elif command == "read":
result = read_pin(pin)
print(result)
sys.exit(0 if result >= 0 else 1)
elif command == "set":
if len(sys.argv) < 4:
print("Usage: python3 gpio_interface.py set <pin> <state>")
sys.exit(1)
state = int(sys.argv[3])
if state not in [0, 1]:
print("Invalid state. Use 0 for LOW or 1 for HIGH.")
sys.exit(1)
result = set_pin_state(pin, state)
sys.exit(result)
else:
print("Invalid command. Use 'toggle', 'read', or 'set'.")
sys.exit(1)
if __name__ == "__main__":
main()