258 lines
8.8 KiB
C++
258 lines
8.8 KiB
C++
|
#include <stdio.h>
|
||
|
#include "pico/stdlib.h"
|
||
|
#include "hardware/uart.h"
|
||
|
#include "hardware/gpio.h"
|
||
|
#include "hardware/spi.h"
|
||
|
#include "pico_wireless.hpp"
|
||
|
#include "secrets.h"
|
||
|
|
||
|
#include <vector>
|
||
|
#include <algorithm>
|
||
|
|
||
|
#define HTTP_PORT 80
|
||
|
#define HTTP_REQUEST_BUF_SIZE 2048
|
||
|
|
||
|
#define DNS_CLOUDFLARE IPAddress(1, 1, 1, 1)
|
||
|
#define DNS_GOOGLE IPAddress(8, 8, 8, 8)
|
||
|
#define USE_DNS DNS_CLOUDFLARE
|
||
|
|
||
|
#define HTTP_REQUEST_DELAY 30 // Seconds between requests
|
||
|
#define HTTP_REQUEST_HOST "api.thingspeak.com"
|
||
|
#define HTTP_REQUEST_PATH "/channels/1417/field/2/last.txt"
|
||
|
#define HTTP_RESPONSE_BUF_SIZE 1024
|
||
|
|
||
|
using namespace pimoroni;
|
||
|
|
||
|
PicoWireless wireless;
|
||
|
uint8_t r, g, b;
|
||
|
uint8_t response_buf[HTTP_RESPONSE_BUF_SIZE];
|
||
|
typedef void(*http_handler)(unsigned int status_code, std::vector<std::string_view> response_head, std::vector<std::string_view> esponse_body);
|
||
|
|
||
|
enum HTTP_REQUEST_STATUS {
|
||
|
HTTP_REQUEST_OK = 0,
|
||
|
HTTP_REQUEST_TIMEOUT,
|
||
|
HTTP_REQUEST_RESPONSE_INVALID,
|
||
|
HTTP_REQUEST_RESPONSE_UNHANDLED,
|
||
|
HTTP_REQUEST_CONNECTION_FAILED,
|
||
|
HTTP_REQUEST_RESPONSE_OVERFLOW,
|
||
|
HTTP_REQUEST_NO_RESPONSE
|
||
|
};
|
||
|
|
||
|
std::vector<std::string_view> split(std::string_view str, std::string delim="\r\n") {
|
||
|
std::vector<std::string_view> result;
|
||
|
size_t offset = 0;
|
||
|
while (offset < str.size()) {
|
||
|
const auto pos = str.find_first_of(delim, offset);
|
||
|
// Emit an empty view even if two adjacent delimiters are found
|
||
|
// this ensurs the HTTP "blank line" start of content is found
|
||
|
result.emplace_back(str.substr(offset, pos - offset));
|
||
|
if (pos == std::string_view::npos) break;
|
||
|
offset = pos + delim.length();
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
uint32_t millis() {
|
||
|
return to_us_since_boot(get_absolute_time()) / 1000;
|
||
|
}
|
||
|
|
||
|
bool wifi_connect(std::string network, std::string password, IPAddress dns_server, uint32_t timeout=10000) {
|
||
|
printf("Connecting to %s...\n", network.c_str());
|
||
|
wireless.wifi_set_passphrase(network, password);
|
||
|
|
||
|
uint32_t t_start = millis();
|
||
|
|
||
|
while(millis() - t_start < timeout) {
|
||
|
if(wireless.get_connection_status() == WL_CONNECTED) {
|
||
|
printf("Connected!\n");
|
||
|
wireless.set_dns(1, dns_server, 0);
|
||
|
return true;
|
||
|
}
|
||
|
wireless.set_led(255, 0, 0);
|
||
|
sleep_ms(500);
|
||
|
wireless.set_led(0, 0, 0);
|
||
|
sleep_ms(500);
|
||
|
printf("...\n");
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/* Basic function to connect to a client IP:PORT and poll for an established connection */
|
||
|
bool connect(IPAddress host_address, uint16_t port, uint8_t client_sock, uint32_t timeout = 1000) {
|
||
|
wireless.start_client(host_address, port, client_sock, TCP_MODE);
|
||
|
|
||
|
uint32_t t_start = millis();
|
||
|
|
||
|
while(millis() - t_start < timeout) {
|
||
|
uint8_t state = wireless.get_client_state(client_sock);
|
||
|
if(state == ESTABLISHED) return true;
|
||
|
sleep_ms(100);
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/* Basic DNS lookup */
|
||
|
IPAddress dns_lookup(std::string request_host) {
|
||
|
IPAddress host_address(0, 0, 0, 0);
|
||
|
printf("DNS lookup for: %s\n", request_host.c_str());
|
||
|
if(!wireless.get_host_by_name(request_host.c_str(), host_address)) {
|
||
|
printf("DNS lookup failed!\n");
|
||
|
}
|
||
|
return host_address;
|
||
|
}
|
||
|
|
||
|
/* This is pretty much the simplest HTTP request function I could get away with.
|
||
|
It accepts a client socket ID, IP address, hostname, request path and handler function,
|
||
|
and calls the handler with the status code (only 200 & 404 at the moment), head
|
||
|
and body as `std::vector<std::string_view>` into the underlying buffer.
|
||
|
*/
|
||
|
HTTP_REQUEST_STATUS http_request(uint8_t client_sock, IPAddress host_address, uint16_t port, std::string request_host, std::string request_path, http_handler handler, uint32_t timeout = 1000) {
|
||
|
if(!connect(host_address, port, client_sock)) {
|
||
|
printf("Connection failed!\n");
|
||
|
return HTTP_REQUEST_CONNECTION_FAILED;
|
||
|
}
|
||
|
|
||
|
// HTTP request to grab our API endpoint
|
||
|
const std::string http_request = "GET " + request_path + " HTTP/1.1\r\n\
|
||
|
Host: " + request_host + "\r\n\
|
||
|
Connection: close\r\n\r\n";
|
||
|
|
||
|
// Clear the response buffer
|
||
|
memset(response_buf, 0, HTTP_RESPONSE_BUF_SIZE);
|
||
|
|
||
|
wireless.send_data(client_sock, (const uint8_t *)http_request.data(), http_request.length());
|
||
|
|
||
|
uint16_t response_length = 0;
|
||
|
uint16_t avail_length = 0;
|
||
|
uint32_t t_start = millis();
|
||
|
|
||
|
// Keep receiving data until our designated timeout
|
||
|
// There's no guarantee that `wireless.avail_data` will have the *whole* response in one shot
|
||
|
// and I *really* don't want to parse the HTTP response for a `Content-Length` header.
|
||
|
while(millis() - t_start < timeout) {
|
||
|
sleep_ms(50);
|
||
|
avail_length = wireless.avail_data(client_sock);
|
||
|
if(avail_length > 0) break;
|
||
|
}
|
||
|
|
||
|
// Read the full response
|
||
|
// Sometimes the bytes read is less than the bytes we request, so loop until we get the data we expect
|
||
|
while(response_length < avail_length) {
|
||
|
uint16_t read_length = avail_length; // Request the full buffer
|
||
|
wireless.get_data_buf(client_sock, response_buf + response_length, &read_length);
|
||
|
response_length += read_length; // Increment the response_length by the amount we actually read
|
||
|
|
||
|
// Also check for timeouts here, too
|
||
|
if(millis() - t_start >= timeout) break;
|
||
|
}
|
||
|
|
||
|
// Explicitly stop our client, and don't leave it dangling!
|
||
|
wireless.stop_client(client_sock);
|
||
|
|
||
|
// Bail if we timed out.
|
||
|
if(millis() - t_start >= timeout) return HTTP_REQUEST_TIMEOUT;
|
||
|
|
||
|
if(response_length > 0) {
|
||
|
std::vector<std::string_view> response = split(std::string_view((char *)response_buf, response_length));
|
||
|
std::vector<std::string_view> response_body;
|
||
|
uint32_t status_code = 0;
|
||
|
|
||
|
// Bail early on an invalid HTTP request
|
||
|
if(response[0].compare(0, 8, "HTTP/1.1") != 0) return HTTP_REQUEST_RESPONSE_INVALID;
|
||
|
|
||
|
// Scan for the blank line indicating content start
|
||
|
auto body_start = std::find(response.begin(), response.end(), "");
|
||
|
|
||
|
// Split the body from the head (ow!)
|
||
|
if(body_start != response.end()) {
|
||
|
response_body = std::vector<std::string_view>(body_start + 1, response.end());
|
||
|
response = std::vector<std::string_view>(response.begin(), body_start);
|
||
|
}
|
||
|
|
||
|
// Parse out the HTTP status code
|
||
|
status_code = std::stoul(std::string(response[0].substr(9, 12)), nullptr);
|
||
|
|
||
|
if(status_code != 0) {
|
||
|
handler(status_code, response, response_body);
|
||
|
return HTTP_REQUEST_OK;
|
||
|
}
|
||
|
|
||
|
return HTTP_REQUEST_RESPONSE_UNHANDLED;
|
||
|
}
|
||
|
|
||
|
return HTTP_REQUEST_NO_RESPONSE;
|
||
|
}
|
||
|
|
||
|
/* As above, but does DNS resolving for us, probably don't use this... */
|
||
|
int http_request(uint8_t client_sock, std::string request_host, uint16_t port, std::string request_path, http_handler handler, uint32_t timeout = 1000) {
|
||
|
IPAddress host_address = dns_lookup(request_host);
|
||
|
return http_request(client_sock, host_address, port, request_host, request_path, handler, timeout);
|
||
|
}
|
||
|
|
||
|
int main() {
|
||
|
stdio_init_all();
|
||
|
|
||
|
wireless.init();
|
||
|
sleep_ms(500);
|
||
|
|
||
|
printf("Firmware version Nina %s\n", wireless.get_fw_version());
|
||
|
|
||
|
if(!wifi_connect(NETWORK, PASSWORD, USE_DNS)) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
g = 255;
|
||
|
wireless.set_led(r, g, b);
|
||
|
|
||
|
// Get a free client socket
|
||
|
uint8_t client_sock = wireless.get_socket();
|
||
|
|
||
|
// Be a good DNS citizen and cache our lookup
|
||
|
IPAddress host_address = dns_lookup(HTTP_REQUEST_HOST);
|
||
|
|
||
|
while(1) {
|
||
|
printf("Requesting: %s\n", HTTP_REQUEST_PATH);
|
||
|
|
||
|
HTTP_REQUEST_STATUS status = http_request(client_sock, host_address, HTTP_PORT, HTTP_REQUEST_HOST, HTTP_REQUEST_PATH, [](
|
||
|
unsigned int status_code,
|
||
|
std::vector<std::string_view> response_head,
|
||
|
std::vector<std::string_view> response_body) {
|
||
|
// Check for valid status
|
||
|
if(status_code != 200) return;
|
||
|
// Check for empty body
|
||
|
if(response_body.size() == 0) return;
|
||
|
// Check for our 7 chars "#000000"
|
||
|
if(response_body[0].length() != 7) return;
|
||
|
// Check for at least a *hopefully* valid hex colour
|
||
|
if(response_body[0].compare(0, 1, "#") != 0) return;
|
||
|
|
||
|
// Convert the hex colour to an unsigned int
|
||
|
uint32_t rgb = std::stoul(std::string(response_body[0].substr(1)), nullptr, 16);
|
||
|
|
||
|
// Unpack to RGB
|
||
|
r = (rgb >> 16) & 0xff;
|
||
|
g = (rgb >> 8) & 0xff;
|
||
|
b = (rgb >> 0) & 0xff;
|
||
|
printf("RGB: %i %i %i\n", r, g, b);
|
||
|
wireless.set_led(r, g, b);
|
||
|
});
|
||
|
|
||
|
if(status == HTTP_REQUEST_NO_RESPONSE) {
|
||
|
printf("No response :(\n");
|
||
|
}
|
||
|
|
||
|
if(status == HTTP_REQUEST_TIMEOUT) {
|
||
|
printf("Request timed out :(\n");
|
||
|
}
|
||
|
|
||
|
if(status == HTTP_REQUEST_RESPONSE_UNHANDLED) {
|
||
|
// Something unexpected happened!
|
||
|
}
|
||
|
|
||
|
sleep_ms(HTTP_REQUEST_DELAY * 1000); // Sensible delay
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|