making interfaces consistent

This commit is contained in:
Ben Menking
2026-05-07 14:41:36 -04:00
parent d59f61d764
commit fb807f42c6
5 changed files with 127 additions and 61 deletions
+3 -1
View File
@@ -44,6 +44,8 @@ class CoreParser {
return self::parseBinaryResponse($payload); return self::parseBinaryResponse($payload);
case CoreProtocol::RESP_CODE_BATT_AND_STORAGE: case CoreProtocol::RESP_CODE_BATT_AND_STORAGE:
return self::parseBatteryAndStorage($payload); return self::parseBatteryAndStorage($payload);
case CoreProtocol::RESP_CODE_NO_MORE_MESSAGES:
return (object)['code'=>CoreProtocol::RESP_CODE_NO_MORE_MESSAGES];
default: default:
return $payload; return $payload;
} }
@@ -170,7 +172,7 @@ class CoreParser {
$data->code = ord($payload[0]); $data->code = ord($payload[0]);
$data->snr = ord($payload[1]) / 4; $data->snr = ord($payload[1]) / 4;
$data->rssi = ord($payload[2]); $data->rssi = (ord($payload[2]) >= 128) ? (ord($payload[2]) - 256): ord($payload[2]);
$data->log = base64_encode(substr($payload, 3, strlen($payload))); $data->log = base64_encode(substr($payload, 3, strlen($payload)));
return $data; return $data;
+2
View File
@@ -179,6 +179,8 @@ class CoreProtocol {
const RESP_CODE_SELF_INFO = 5; const RESP_CODE_SELF_INFO = 5;
const RESP_CODE_SENT = 6; const RESP_CODE_SENT = 6;
const RESP_CODE_CURR_TIME = 9; const RESP_CODE_CURR_TIME = 9;
const RESP_CODE_NO_MORE_MESSAGES = 10;
const RESP_CODE_EXPORT_CONTACT = 11;
const RESP_CODE_BATT_AND_STORAGE = 12; const RESP_CODE_BATT_AND_STORAGE = 12;
const RESP_CODE_DEVICE_INFO = 13; const RESP_CODE_DEVICE_INFO = 13;
+110 -43
View File
@@ -2,11 +2,8 @@
namespace Menking\Meshcore; namespace Menking\Meshcore;
use Menking\Meshcore\Model\BatteryStorageResponse;
use Menking\Meshcore\Model\BinaryResponse;
use Menking\Meshcore\Model\DeviceInfoResponse;
use Menking\Meshcore\Model\Response;
use RuntimeException; use RuntimeException;
use Menking\Meshcore\Model\BinaryResponse;
class Meshcore { class Meshcore {
private $serial; private $serial;
@@ -22,6 +19,11 @@ class Meshcore {
} }
} }
/**
*
* @return Meshcore
* @throws RuntimeException
*/
public static function getInstance(): Meshcore { public static function getInstance(): Meshcore {
if( self::$instance == null && Environment::getDevice() == null ) { if( self::$instance == null && Environment::getDevice() == null ) {
throw new RuntimeException("Please initialize Meshcore by calling Environment::configure()"); throw new RuntimeException("Please initialize Meshcore by calling Environment::configure()");
@@ -34,11 +36,11 @@ class Meshcore {
} }
/** /**
* Initialize the connection. Needed for most commands to succeed.
* *
* @return true|array{code: int, pub_key: string, type: int, flags: int, out_path_len: int, out_path: string, contact_name: string, last_advert_timestamp: string, gps_lat: int|float, gps_lon: int|float, lastmod: string}|array{code: int, sent_flood: int, pending_login: mixed, est_timeout: mixed}|string * @param string $app_name
* @return mixed
*/ */
public function appStart(string $app_name) { public function appStart(string $app_name): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_APP_START) . chr(0x00) . " " . $app_name); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_APP_START) . chr(0x00) . " " . $app_name);
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
@@ -47,9 +49,9 @@ class Meshcore {
/** /**
* *
* @return BatteryStorageResponse * @return mixed
*/ */
public function getBatteryAndStorage(): BatteryStorageResponse { public function getBatteryAndStorage(): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_BATT_AND_STORAGE)); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_BATT_AND_STORAGE));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
@@ -66,9 +68,9 @@ class Meshcore {
/** /**
* *
* @return DeviceInfoResponse * @return mixed
*/ */
public function getDeviceInfo(): DeviceInfoResponse { public function getDeviceInfo(): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_DEVICE_QUERY) . 0x03); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_DEVICE_QUERY) . 0x03);
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
@@ -78,25 +80,24 @@ class Meshcore {
/** /**
* *
* @param int $channel_idx * @param int $channel_idx
* @return array{response_code: int, channel_idx: int, channel: string[], channel_secret: string} * @return mixed
*/ */
public function getChannel(int $channel_idx) { public function getChannel(int $channel_idx): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_CHANNEL) . chr($channel_idx)); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_CHANNEL) . chr($channel_idx));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
public function setRadioParams(float $freq, float $bw, int $sf, int $cr) { /**
/* *
$params = [ * @param float $freq
'freq'=>910.525, * @param float $bw
'bw'=>62.5, * @param int $sf
'sf'=>7, * @param int $cr
'cr'=>5, * @return mixed
]; */
*/ public function setRadioParams(float $freq, float $bw, int $sf, int $cr): mixed {
$payload = chr(CoreProtocol::CMD_SET_RADIO_PARAMS) $payload = chr(CoreProtocol::CMD_SET_RADIO_PARAMS)
. pack('V', $freq * 1000) . pack('V', $freq * 1000)
. pack('V', $bw * 1000) . pack('V', $bw * 1000)
@@ -110,6 +111,10 @@ class Meshcore {
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
/**
*
* @return array
*/
public function getContacts(): array { public function getContacts(): array {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_CONTACTS)); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_CONTACTS));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
@@ -128,18 +133,35 @@ class Meshcore {
return $contacts; return $contacts;
} }
public function getNextMessage(?string $tag = null) { /**
*
* @param null|string $tag
* @return mixed
*/
public function getNextMessage(?string $tag = null): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SYNC_NEXT_MESSAGE)); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SYNC_NEXT_MESSAGE));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
if( ord($response[0]) == 0x0a ) { $data = CoreParser::parseResponse($response);
return [];
if( $data->code == 10 ) { // RESP_CODE_NO_MORE_MESSAGES
return null;
} }
return CoreParser::parseResponse($response); if( !is_null($tag) && $data->tag != $tag ) {
return null;
}
return $data;
} }
public function sendChannelTxtMessage(string $message, int $channel_idx): bool { /**
*
* @param string $message
* @param int $channel_idx
* @return mixed
*/
public function sendChannelTxtMessage(string $message, int $channel_idx): mixed {
$payload = chr(CoreProtocol::CMD_SEND_CHANNEL_TXT_MSG) . chr(0x00) . chr($channel_idx) . pack('V', time()) . "$message\0"; $payload = chr(CoreProtocol::CMD_SEND_CHANNEL_TXT_MSG) . chr(0x00) . chr($channel_idx) . pack('V', time()) . "$message\0";
CoreProtocol::writeFrame($this->serial, $payload); CoreProtocol::writeFrame($this->serial, $payload);
@@ -148,14 +170,22 @@ class Meshcore {
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
public function getDeviceTime(){ /**
*
* @return mixed
*/
public function getDeviceTime(): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_DEVICE_TIME)); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_GET_DEVICE_TIME));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
public function setDeviceTime() { /**
*
* @return mixed
*/
public function setDeviceTime(): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SET_DEVICE_TIME) . pack('V', time())); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SET_DEVICE_TIME) . pack('V', time()));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
@@ -166,9 +196,9 @@ class Meshcore {
* *
* @param string $contact_pk * @param string $contact_pk
* @param string $password * @param string $password
* @return true|array{code: int, pub_key: string, type: int, flags: int, out_path_len: int, out_path: string, contact_name: string, last_advert_timestamp: string, gps_lat: int|float, gps_lon: int|float, lastmod: string}|array{code: int, sent_flood: int, pending_login: mixed, est_timeout: mixed}|string * @return mixed
*/ */
public function login(string $contact_pk, string $password) { public function login(string $contact_pk, string $password): mixed {
$payload = chr(CoreProtocol::CMD_LOGIN) . str_pad(base64_decode($contact_pk), CoreProtocol::PUB_KEY_SIZE, "\0") . $password . chr(0x00); $payload = chr(CoreProtocol::CMD_LOGIN) . str_pad(base64_decode($contact_pk), CoreProtocol::PUB_KEY_SIZE, "\0") . $password . chr(0x00);
CoreProtocol::writeFrame($this->serial, $payload); CoreProtocol::writeFrame($this->serial, $payload);
@@ -186,17 +216,21 @@ class Meshcore {
/** /**
* *
* @param string $contact_pk * @param string $contact_pk
* @return true|array{code: int, pub_key: string, type: int, flags: int, out_path_len: int, out_path: string, contact_name: string, last_advert_timestamp: string, gps_lat: int|float, gps_lon: int|float, lastmod: string}|array{code: int, sent_flood: int, pending_login: mixed, est_timeout: mixed}|string * @return mixed
*/ */
public function disconnect(string $contact_pk) { public function disconnect(string $contact_pk): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_LOGOUT) . base64_decode($contact_pk) . "\0"); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_LOGOUT) . base64_decode($contact_pk) . "\0");
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
/**
public function statusRequest(string $contact_pk ) { *
* @param string $contact_pk
* @return mixed
*/
public function statusRequest(string $contact_pk ): mixed {
$payload = chr(CoreProtocol::CMD_SEND_STATUS_REQ) . base64_decode($contact_pk); $payload = chr(CoreProtocol::CMD_SEND_STATUS_REQ) . base64_decode($contact_pk);
CoreProtocol::writeFrame($this->serial, $payload); CoreProtocol::writeFrame($this->serial, $payload);
@@ -205,6 +239,11 @@ class Meshcore {
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
/**
*
* @param string $contact_pk
* @return bool
*/
public function connected(string $contact_pk): bool { public function connected(string $contact_pk): bool {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_HAS_CONNECTION) . base64_decode($contact_pk) . "\0"); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_HAS_CONNECTION) . base64_decode($contact_pk) . "\0");
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
@@ -212,35 +251,63 @@ class Meshcore {
return ord($response[0]) != 0x01; return ord($response[0]) != 0x01;
} }
public function getTelemetryRequest(string $contact_pk) { /**
*
* @param string $contact_pk
* @return mixed
*/
public function getTelemetryRequest(string $contact_pk): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SEND_TELEMETRY_REQ) . base64_decode($contact_pk)); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SEND_TELEMETRY_REQ) . base64_decode($contact_pk));
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
public function sendAnonRequest(string $contact_pk) { /**
*
* @param string $contact_pk
* @return mixed
*/
public function sendAnonRequest(string $contact_pk): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SEND_ANON_REQ) . base64_decode($contact_pk) . "\0"); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SEND_ANON_REQ) . base64_decode($contact_pk) . "\0");
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
public function sendBinaryRequest(string $request): BinaryResponse { /**
*
* @param string $request
* @return mixed
*/
public function sendBinaryRequest(string $request): mixed {
CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SEND_BINARY_REQ) . $request); CoreProtocol::writeFrame($this->serial, chr(CoreProtocol::CMD_SEND_BINARY_REQ) . $request);
$response = CoreProtocol::readFrame($this->serial); $response = CoreProtocol::readFrame($this->serial);
return CoreParser::parseResponse($response); return CoreParser::parseResponse($response);
} }
public function poll(int $timeout_ms = 5000): void { /**
*
* @param string $tag
* @param int $timeout_ms
* @return mixed
*/
public function pollForMessage(string $tag, int $timeout_ms = 5000): mixed {
$mark = time(); $mark = time();
do { do {
$response = CoreProtocol::readFrame($this->serial); $msg = self::getNextMessage();
echo json_encode(CoreParser::parseResponse($response));
if( $msg instanceof BinaryResponse && $msg->tag == $tag ) {
return $msg;
}
usleep(500000);
} }
while( (time() - $mark) < $timeout_ms ) ; //ord($response[0]) == 0x01 && ord($response[1]) == 0xff); while( (time() - $mark) < $timeout_ms );
return false;
} }
} }
+8 -4
View File
@@ -24,10 +24,14 @@ $contact = $contacts[$choice - 1];
echo "Using {$contact['contact_name']}\n"; echo "Using {$contact['contact_name']}\n";
print_r($mc->login($contact['pub_key'], '')); print_r($mc->login($contact['pub_key'], readline('Please enter the password: ')));
$ts = $mc->statusRequest($contact['pub_key']); $result = $mc->statusRequest($contact['pub_key']);
print_r($result);
exit;
print_r($ts); $msg = $mc->pollForMessage($result->tag);
$mc->poll(); print_r($msg);
echo "Complete\n";
+4 -13
View File
@@ -36,20 +36,11 @@ if( !$mc->connected($contact['pub_key']) ) {
$req = base64_decode($contact['pub_key']) . chr(CoreProtocol::BINREQ_TELEMETRY); $req = base64_decode($contact['pub_key']) . chr(CoreProtocol::BINREQ_TELEMETRY);
$result = $mc->sendBinaryRequest($req); $result = $mc->sendBinaryRequest($req);
echo "Tag: {$result['tag']}\n";
$timeout = 0; echo "Tag: {$result->tag}\n";
while($timeout++ < 6) { $msg = $mc->pollForMessage($result->tag);
echo "timeout: $timeout\n";
$resp = $mc->getSyncNextMessage(); print_r($msg);
if( !empty($resp) ) {
print_r($resp);
if( $resp['code'] == 0x8c) {
exit;
}
}
sleep(1);
}
echo "Complete\n"; echo "Complete\n";