ESP32 HTTPS POST JSON 到 AWS

ESP32 HTTPS POST JSON to AWS

我正在尝试 post 通过 HTTPS post 将一些数据发送到 AWS,但似乎无法到达那里。

正在将主机设置为服务器地址和 /prod 路径 该方法还不需要身份验证。 当我连接时它只是冻结并且没有响应, 尽管说它已连接。

还尝试不使用路径并设置主机的完整路径

不太确定为什么会这样,这是我尝试过的两种方法:

    void postToAmazonSecure(String jsonToSend){
  WiFiClientSecure client;
  
  Serial.print("connecting to : '");
  Serial.print(emonDataAPI);
  Serial.println("'");
  Serial.println(emonDataAPI.c_str());
  client.connect(emonDataAPI.c_str(), 443);
  
  Serial.print("requesting URL: '");
  Serial.print(emonDataAPI);
  Serial.println("'");
  String requestString = String("POST ") + emonDataAPIPath + " HTTP/1.1\r\n" +
    "Host: " + emonDataAPI + "\r\n" +
    "Connection: close\r\n" +
    "Content-Type: application/json" +
    //"Authorization: Bearer " + authorization_code + "\r\n" +
    "Content-Length: " + jsonToSend.length() + "\r\n" +
    "\r\n" +
    jsonToSend + "\r\n";
    Serial.println(requestString);
  client.print(requestString);
  
  Serial.println("request sent");
  unsigned long timeout = millis();
  while (client.available() == 0) {
    if (millis() - timeout > 5000) {
      Serial.println(">>> Client Timeout !");
      client.stop();
      return;
    }
}
}

void postSecure(String jsonToSend){
 WiFiClientSecure client;

 Serial.print("connect: "); Serial.println(emonDataAPI);
 while ( ! client.connect(emonDataAPI.c_str(), 443)) {
    Serial.print(".");
 }
 Serial.println("Connected");
 String msg = "POST " + emonDataAPIPath + " HTTP/1.1\r\n"
                "Host: " + emonDataAPI + "\r\n"
                "Content-Type: application/json\r\n"
                "Content-Length: " + jsonToSend.length() + "\r\n"
                "\r\n" + jsonToSend;
                
 client.print(msg);
 Serial.print(msg);
 
 Serial.print("\n*** Request sent, receiving response...");
 while (!!!client.available()) {
    delay(50);
 Serial.print(".");
  }
  
Serial.println();
Serial.println("Got response");  
  while(client.available()){
  Serial.write(client.read());
  }
Serial.println(); Serial.println("closing connection");
  client.stop();
}

您忘记了 Content-Type 行末尾的 \r\n。应该是:

"Content-Type: application/json\r\n" +

您也没有向 WiFiClientSecure 提供指纹或证书,以便它可以验证您正在连接的服务器。这也会使您的代码停止工作。您应该为保护您尝试连接的端点的证书链提供根证书:

const char* test_root_ca= \
     "-----BEGIN CERTIFICATE-----\n" \
...
     "-----END CERTIFICATE-----\n";


...

  client.setCACert(test_root_ca);
  client.connect(emonDataAPI.c_str(), 443);

您可以在 WiFiClientSecure library examples 中找到更完整的示例。

您还可以查看 README for WiFiClientSecure 了解更多详细信息。

没有根证书WiFiClientSecure 无法验证您要连接的服务器的身份。 Windows、Linux 和 macOS 系统都会自动处理这个问题;这样做的开销超出了 ESP32 的操作环境可以处理的范围。

您可以告诉 WiFiClientSecure 不要费心验证服务器,但这样做也是在说“我并不真正关心安全性”。您可能会为了调试而这样做,但将其留在实际使用的代码中是一种极其糟糕和危险的做法。

  client.setInsecure();
  client.connect(emonDataAPI.c_str(), 443);

这些类型的错误是使用 HTTPClient 这样的库几乎总是比滚动自己的 HTTP(S) 实现更好的主要原因。