2022年2月27日日曜日
技適マーク付きESP32-CAMを入手
2022年2月26日土曜日
ESP32のdeep sleepは寝起きが悪い
技適品に換装したESP32-CAMと焦電センサを使ってLINEに通知する防犯カメラですが、色々とスケッチをいじって遊んでいます。
大まかな流れは、普段はdeep sleepしていて、焦電センサが人を検知したタイミングでwake upして、写真撮影し、micrSDに保存、LINE notifyへの送信するというものです。LINE notifyへの画像送付は、頻度に制限があるようなので90秒待ってからdeep sleepします。これで次にセンサが検知したら再びwake upして同じ処理を行います。
なお、90秒待つ間にもセンサが検知したら写真を撮影してmicroSDへ保存します。こちらは最短1秒毎にして、センサの前に人がいれば連続して撮影されます。
数日家の中に置いて試行運用してみましたが、どうもLINEに送られてくる写真に人物が映りません。microSDには人が映ったものも残っています。
ESP32のdeep sleep & wake upは、resetしたのと同じようにESP32の起動から動くのでアプリ処理が動くまで結構タイムラグがあるようです。
millis()関数を使い、トリガがかかってからsetup処理が始まる時間と、カメラの設定を行って撮影するまでの時間を調べてみました。
setupが動くまでに0.8秒程かかっており、さらにカメラの設定などを行い撮影するまでに約0.7秒必要で、実際の撮影はセンサが検知してから1.5秒後になっている模様。これでは、シャッタチャンスを逃しますね。なお、Serial.printでモニタしているので、その処理にも若干時間がかかっているとは思います。
ということで、deep sleepはやめて、light sleepにしようと思っています。deep sleepならバッテリ運用も可能かと思っていましたがちょっと無理そうです。
なお、LINE notifyへ写真を送るスケッチはgithhubの以下を流用さてせ頂いています。
https://github.com/fustyles/Arduino/tree/master/ESP32-CAM_Linenotify
※そのままビルドするとおねえさんの写真が送られてくる(URLが書かれている)ので注意。
2022年2月23日水曜日
ESP32-CAMを技適ESP32へ換装する動画
2022年2月20日日曜日
カメラモジュールESP32-CAMを技適取得済みESP32に換装
2022年2月15日火曜日
電子工作用のmicroSD購入
2022年2月14日月曜日
MacBookの乗り換え、再び
前回の失敗を踏まえてHDD経由で移行 しようと、現在Airのバックアップ中です。今夜寝る前に新MacBook Proでリストアを始めれば明日の朝には終わっていることを願っています。
2022年2月13日日曜日
焦電人感センサでLINEへ通知する防犯センサ
電源を入れて30秒は不安定とのことで30秒待ってから監視を開始しましたが、どうも感度がよいのかノイズがあるのか、プルダウン抵抗が大きすぎたのか中々Lowレベルで安定しなかったので、スケッチ中にLowになるまで待つ処理を入れたらOKになりました。もう少し調整したほうがよいか。
LINE notify サービスは以下参照。
2022年2月12日土曜日
LINEダッシュボタン、3Dプリンタ印刷完了を通知する機能の完成
内部はこんな感じです。暗号化したWiFiのSSIDやパスワード、LINE notifyのトークン、およびポストするメッセージを格納したmicroSDはケースの蓋を開けないと取り外しできないようにしました。写真左側は、フォトリフレクタを収めたセンサ部です。
回路図です。三端子レギュレータは1Aのものを使いましたが、500mA取れれば大丈夫なようです。
//---------------------------------------------------------------
//                                                2022.02.12 naka
// LINE Dash Button(フォトリフレクタ版)
//
// ・WiFi ssid,password,LINE notify tokenは、暗号化してmicroSDに格納
//  ファイル名:setup.txt
//     暗号化したWiFi ssid, password, LINE notify tokenをそれぞれ1行ずつ
//     順に記載し、4行目、5行目に送るメッセージを記載。
//  (例)
//    &D/)%X/C04S+#T'?+USM
//    *%/O-J_F$##5)CK`(%
//    #VGN-Z_*-FCG+VK1/E;M<47E1UCJ.C/`-U(S#F?C06CV)%C_0$$E8W$89(0[5(U
//    動作を開始しました。
//    ボタンが押されました。
//---------------------------------------------------------------
#include <WiFi.h>
#include <WiFiClientSecure.h>
#include <SD.h>
#define SD_CS 16 // SD card chip select
#define SW    GPIO_NUM_13
#define LED_G 25
#define LED_R 27
#define ON  1
#define OFF 0
#define FILENAME "/setup.txt"
// ssid,pw,tokenを暗号化したときのキー
char* crypt_key = "Open Sesame";
// LINE Notify
const char* host  = "notify-api.line.me";
char ssid[64],password[64],token[64],message1[128],message2[128];
int sw_newest;
void setup()
{
  pinMode(SW,INPUT_PULLUP);
  pinMode(LED_G,OUTPUT);
  pinMode(LED_R,OUTPUT);
  led_green(OFF);
  led_red(OFF);
  sw_newest = digitalRead(SW);
  if (read_setup_file(FILENAME)) {
    blink_red(10);
    esp_sleep_enable_ext0_wakeup(SW, LOW);
    esp_deep_sleep_start();
  }
  // GPIO割り込みピン設定
  gpio_wakeup_enable(SW, GPIO_INTR_LOW_LEVEL);
  // 動作確認のために動作開始メッセージを送る
  wifi_connect();          // WiFi接続
  send_msg2line(message1); // LINE notifyに1行目のメッセージを送る
  wifi_disconnect();       // WiFi接続解除
  led_green(ON);
  // GPIO割り込みでのsleep復帰を有効化
  esp_sleep_enable_gpio_wakeup();
}
void loop(){
  // スリープ開始(復帰はこの後ろ)
  esp_light_sleep_start();
  
  wifi_connect();          // WiFi接続
  send_msg2line(message2); // LINE notifyに2行目のメッセージを送る
  wifi_disconnect();       // WiFi接続解除
  led_green(ON);
  while(!digitalRead(SW));
  delay(100);              // チャタリング対策
}
void led_green(int onoff) {
  if (onoff==ON) digitalWrite(LED_G,LOW);
  else           digitalWrite(LED_G,HIGH);
}
void led_red(int onoff) {
  if (onoff==ON) digitalWrite(LED_R,LOW);
  else           digitalWrite(LED_R,HIGH);
}
void wifi_connect() {
    WiFi.begin(ssid, password);
    int onoff = 0;
    while (WiFi.status() != WL_CONNECTED) {
        delay(500);
        if (onoff==0) led_green(ON);
        else          led_green(OFF);
        onoff ^= 1;
    }
    // 以下5行は、普通は不要なハズ
    IPAddress gateway(192,168,0,1);
    IPAddress subnet(255,255,255,0);
    IPAddress dns1(8, 8, 8, 8); // Google DNS primery server
    IPAddress dns2(8, 8, 4, 4); // Google DNS secondary server
    WiFi.config(WiFi.localIP(),gateway,subnet,dns1,dns2);    
    led_green(ON);
}
void wifi_disconnect() {
    WiFi.disconnect(true);
    led_green(OFF);
}
void send_msg2line(char* message) {
  WiFiClientSecure client;
  led_red(ON);
  client.setInsecure();
  if (!client.connect(host, 443)) {
    blink_red(10);
    led_red(OFF);
    return;
  }
  
  String query = String("message=") + String(message);
  String request = String("") +
               "POST /api/notify HTTP/1.1\r\n" +
               "Host: " + host + "\r\n" +
               "Authorization: Bearer " + token + "\r\n" +
               "Content-Length: " + String(query.length()) +  "\r\n" + 
               "Content-Type: application/x-www-form-urlencoded\r\n\r\n" +
                query + "\r\n";
  client.print(request);
 
  while (client.connected()) {
    String line = client.readStringUntil('\n');
    if (line == "\r") {
      break;
    }
  }
  
  String line = client.readStringUntil('\n');
  led_red(OFF);
}
void blink_red(int no) {
  int onoff = 1;
  int i;
  for (i=0; i<no*2 ;i++) {
    delay(100);
    if (onoff==0) led_red(ON);
    else          led_red(OFF);
    onoff ^= 1;
  }
}
int read_setup_file(char* fileName) {
  char  buff[128];
  if (!SD.begin(SD_CS, SPI, 24000000)) return 1;
  File setupFile = SD.open(fileName,FILE_READ);
  if (setupFile) {
    readLine(setupFile,buff);
    decrypt(ssid,buff,crypt_key);
    
    readLine(setupFile,buff);
    decrypt(password,buff,crypt_key);
    
    readLine(setupFile,buff);
    decrypt(token,buff,crypt_key);
    
    readLine(setupFile,message1);
    readLine(setupFile,message2);
    setupFile.close();
  }
  return 0;
}
// microSDから1行読み取る
int readLine(File fp, char *buff){
  int i = 0;
  if(!fp.available()) {
    return -1;
  }
  while (fp.available()) {
    char c = fp.read();
    if (c=='\r');
    else if (c=='\n') break;
    else buff[i++] = c;
  }
  buff[i] = 0;
  
  return i;
}
void decrypt( char*a, char*b, char* key) {
  char x[128];
  int i,j,k,m;
  // ascii文字をbinaryに戻す
  for(i=0; b[i]!=0; i++) {
    x[i] = b[i] - (int)'#';
  }
  m = i - 1;
  // 6bit -> 8bit
  j = 0;
  for(i=0; j<m ; i++) {
    switch (i%3) {
      case 0:
        a[i] = (x[j]&0x3F)<<2 | (x[j+1]&0x30)>>4;
        j++;
        break;
      case 1:
        a[i] = (x[j]&0x0F)<<4 | (x[j+1]&0x3C)>>2; 
        j++;
        break;
      case 2:
        a[i] = (x[j]&0x03)<<6 | x[j+1]&0x3F;
        j += 2;
        break;
    }
  }
  m = i;
  
  k = 0;
  int n = strlen(key);
  for (i=0; i<m; i++) {
    a[i] ^= key[k++];
    k %= n;
  }
  a[i]= 0;
}
//---------------------------------------------------------------
// EOF
//---------------------------------------------------------------
2022年2月11日金曜日
LINEダッシュボタン、Windows版暗号化プログラム他
1. フォトリフレクタ固定治具
印刷が完了すると、右上に少し見えているヒートベッドが移動してフォトリフレクタの上に来ます。それをトリガにして、LINEダッシュボタンが印刷完了メッセージを送る仕掛けです。
2.消費電力測定
3. Windows版暗号化プログラム
//----------------------------------------------------------------------
//  '22.02.13 bug fix
//----------------------------------------------------------------------
#include <stdio.h>
#include <string.h>
void encrypt( char*a, char*b, char* key) {
  char x[128];
  int i,j,k,m;
  k = 0;
  int n = strlen(key);
  for (i=0; b[i]!=0; i++) {
    x[i] = b[i]^key[k++];
    k %= n;
  }
  m = i;
  // 8bit -> 6bit
  // 8bit           0          1          2
  //        000000|00, 0000|0000, 00|000000
  // 6bit        0        1        2      3
  j = 0;
  for (i=0; i<m; j++) {
    switch (j%4) {
      case 0:
        a[j] = (x[i]&0xFC)>>2;
        break;
      case 1:
        a[j] = (x[i]&0x03)<<4 | (x[i+1]&0xF0)>>4; 
        i++;
        break;
      case 2:
        a[j] = (x[i]&0x0F)<<2 | (x[i+1]&0xC0)>>6; 
        i++;
        break;
      case 3:
        a[j] = x[i]&0x3F;
        i++; 
        break;
    }
  }
  // ascii文字に変換
  for (i=0; i<j; i++) {
    a[i] += (int)'#';
  }
  a[i] = 0;
  
}
void decrypt( char*a, char*b, char* key) {
  char x[128];
  int i,j,k,m;
  // ascii文字をbinaryに戻す
  for(i=0; b[i]!=0; i++) {
    x[i] = b[i] - (int)'#';
  }
  m = i - 1;
  // 6bit -> 8bit
  j = 0;
  for(i=0; j<m ; i++) {
    switch (i%3) {
      case 0:
        a[i] = (x[j]&0x3F)<<2 | (x[j+1]&0x30)>>4;
        j++;
        break;
      case 1:
        a[i] = (x[j]&0x0F)<<4 | (x[j+1]&0x3C)>>2; 
        j++;
        break;
      case 2:
        a[i] = (x[j]&0x03)<<6 | (x[j+1]&0x3F);
        j += 2;
        break;
    }
  }
  m = i;
  
  k = 0;
  int n = strlen(key);
  for (i=0; i<m; i++) {
    a[i] ^= key[k++];
    k %= n;
  }
  a[i]= 0;
}
int main(int argc, char *argv[]) {
  int i;
  if (argc!=3) {
    printf("Usage: encrypt.exe key_text file_name\n");
    return(0);
  }
  char* key  = argv[1];
  char* file = argv[2];
  char  cryptfile[128];
  strcpy(cryptfile,file);
  strcat(cryptfile,"_crypt.txt");
  FILE *fp = fopen(file,"r");
  if (!fp) {
    printf("input file open error. %s\n",file);
    return(-1);
  }
  FILE *cfp = fopen(cryptfile,"w");
  if (!cfp) {
    printf("output file open error. %s\n",cryptfile);
    return(-1);
  }
  char plain_text[128];
  char encrypted[256];
  char decrypted[256];
  for (i=0;i<3;i++) {
    fgets(plain_text,128,fp);
    strtok(plain_text,"\n\0");
    encrypt(encrypted,plain_text,key);
    fprintf(cfp,"%s\n",encrypted);
    printf("encrypted:%s\n",encrypted);
    decrypt(decrypted,encrypted,key);
    printf("decrypted:%s",decrypted);
  }
  while( fgets(plain_text,128,fp) ) {
    printf("plaintext:%s",plain_text);
    fprintf(cfp,"%s",plain_text);
  }
  
  fclose(cfp);
  fclose(fp);
  printf("\nencrypted file : %s\n",cryptfile);
}
2022年2月9日水曜日
LINEダッシュボタン、sleepとwakeup
void setup()
{
  ... 省略...
  
  // GPIO割り込みピン設定
  gpio_wakeup_enable(SW, GPIO_INTR_LOW_LEVEL);
  // GPIO割り込みでのsleep復帰を有効化
  esp_sleep_enable_gpio_wakeup();
}
void loop(){
  // スリープ開始(復帰はこの後ろ)
  esp_light_sleep_start();
  
  wifi_connect();          // WiFi接続
  send_msg2line(message2); // LINE notifyに2行目のメッセージを送る
  wifi_disconnect();       // WiFi接続解除
  while(!digitalRead(SW));
  delay(100);              // チャタリング対策
}
2022年2月6日日曜日
LINEダッシュボタン、ユニバーサル基板に実装
2022年2月5日土曜日
e-Taxで確定申告を済ませました
2時間ほどで終わりましたが、慣れない言葉や細かい数字を見たりで疲れました。期待していたより少ないですが還付されることになりました。時給換算すると結構よい(^_^) まぁ自動的に徴収されていたものが戻ってくるだけなので、報酬という訳ではないですが。
2022年2月4日金曜日
LINEダッシュボタンで3Dプリント完了通知
3Dプリンタは印刷に数時間とかかかるので、これまでは時々見に行っていました。予想時間が表示されていますがあまり当てにならないし、忘れてしまうこともありました。
ヒートベッドが戻る位置の下にフォトリフレクタを付けています。ひとまず仮なのでマスキングテープで貼り付け。ヒートベッドの裏側は黒くて赤外線が反射しにくいので、アルミテープを貼ってよく反射するようにしました。
実際に使ってみるとすごく便利です。連続していくつも印刷したいときに、ヘッドやヒートベッドが冷める前に次の印刷に取り掛かれます。ただ準備などでうっかりフォトリフレクタを遮るとメッセージが飛ぶので、印刷を開始してから電源を入れます。
思いのほか便利なのでケースに入れて常用にしようと思います。フォトリフレクタを点灯しっぱなしにするので、電池稼働ではなくACアダプタにするつもりです。
2022年2月2日水曜日
LINEダッシュボタン、microSD対応
平文でパスワードやトークンを書くのはリスキーなので簡単な暗号化を行いました。EXOR暗号とシーザー暗号のようなものを組み合わせています。以下のような感じで暗号化されます。
key :Open Sesame plain_ssid :WiFi_AccessPoint_SSID plain_passwd :SSID_password plain_token :LINE_TOKEN_ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789 crypted_ssid :)$GF$Z_5$D#'*D;B*SS#8#SY+%CL crypted_passwd:*%/O-J_F$##5)CK`(# crypted_token :#VGN-Z_*-FCG+VK1/E;M<47E1UCJ.C/`-U(S#F?C06CV)%C_0$$E8W$89(0[5(U
//--------------------------------------------------------------
// 簡単なXOR暗号                                    '22.01.31 naka
//  実行例)
//   key           :Open Sesame
//
//   plain_ssid    :WiFi_AccessPoint_SSID
//   plain_passwd  :SSID_password
//   plain_token   :LINE_TOKEN_ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789
//
//   crypted_ssid  :)$GF$Z_5$D#'*D;B*SS#8#SY+%CL
//   crypted_passwd:*%/O-J_F$##5)CK`(#
//   crypted_token :#VGN-Z_*-FCG+VK1/E;M<47E1UCJ.C/`-U(S#F?C06CV)%C_0$$E8W$89(0[5(U
//
//   decrypt_ssid  :WiFi_AccessPoint_SSID
//   decrypt_passwd:SSID_password
//   decrypt_token :LINE_TOKEN_ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789
//--------------------------------------------------------------
void setup() {
  char crypted_ssid[128],crypted_passwd[128],crypted_token[128];
  char dec_ssid[128],dec_passwd[128],dec_token[128];
  Serial.begin(115200);
  
  char* key          = "Open Sesame";
  char* plain_ssid   = "WiFi_AccessPoint_SSID";
  char* plain_passwd = "SSID_password";
  char* plain_token  = "LINE_TOKEN_ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  encrypt(crypted_ssid,plain_ssid,key);
  encrypt(crypted_passwd,plain_passwd,key);
  encrypt(crypted_token,plain_token ,key);
  decrypt(dec_ssid,crypted_ssid,key);
  decrypt(dec_passwd,crypted_passwd,key);
  decrypt(dec_token,crypted_token,key);
  Serial.printf("key           :%s\n\n",key);
  Serial.printf("plain_ssid    :%s\n",plain_ssid);
  Serial.printf("plain_passwd  :%s\n",plain_passwd);
  Serial.printf("plain_token   :%s\n",plain_token);
  Serial.printf("\n");
  Serial.printf("crypted_ssid  :%s\n",crypted_ssid);
  Serial.printf("crypted_passwd:%s\n",crypted_passwd);
  Serial.printf("crypted_token :%s\n",crypted_token);
  Serial.printf("\n");
  Serial.printf("decrypt_ssid  :%s\n",dec_ssid);
  Serial.printf("decrypt_passwd:%s\n",dec_passwd);
  Serial.printf("decrypt_token :%s\n",dec_token);
  Serial.printf("\n");
  sleep;
}
void loop() {
}
void encrypt( char*a, char*b, char* key) {
  char x[128];
  int i,j,k,m;
  k = 0;
  int n = strlen(key);
  for (i=0; b[i]!=0; i++) {
    x[i] = b[i]^key[k++];
    k %= n;
  }
  m = i;
  // 8bit -> 6bit
  // 8bit           0          1          2
  //        000000|00, 0000|0000, 00|000000        
  // 6bit        0        1        2      3  
  j = 0;
  for (i=0; i<m; j++) {
    switch (j%4) {
      case 0:
        a[j] = (x[i]&0xFC)>>2;
        break;
      case 1:
        a[j] = (x[i]&0x03)<<4 | (x[i+1]&0xF0)>>4; 
        i++;
        break;
      case 2:
        a[j] = (x[i]&0x0F)<<2 | (x[i+1]&0xC0)>>6; 
        i++;
        break;
      case 3:
        a[j] = x[i]&0x3F;
        i++; 
        break;
    }
  }
  // ascii文字に変換
  for (i=0; i<j; i++) {
    a[i] += (int)'#';
  }
  a[i] = 0;
  
}
void decrypt( char*a, char*b, char* key) {
  char x[128];
  int i,j,k,m;
  // ascii文字をbinaryに戻す
  for(i=0; b[i]!=0; i++) {
    x[i] = b[i] - (int)'#';
  }
  m = i - 1;
  // 6bit -> 8bit
  j = 0;
  for(i=0; j<m ; i++) {
    switch (i%3) {
      case 0:
        a[i] = (x[j]&0x3F)<<2 | (x[j+1]&0x30)>>4;
        j++;
        break;
      case 1:
        a[i] = (x[j]&0x0F)<<4 | (x[j+1]&0x3C)>>2; 
        j++;
        break;
      case 2:
        a[i] = (x[j]&0x03)<<6 | x[j+1]&0x3F;
        j += 2;
        break;
    }
  }
  m = i;
  
  k = 0;
  int n = strlen(key);
  for (i=0; i<m; i++) {
    a[i] ^= key[k++];
    k %= n;
  }
  a[i]= 0;
}
回路図とスケッチです。無保証です。
//---------------------------------------------------------------
//                                                2022.02.02 naka
// LINE Dash Button
//
// ・WiFi ssid,password,LINE notify tokenは、暗号化してmicroSDに格納
//  ファイル名:setup.txt
//     暗号化したWiFi ssid, password, LINE notify tokenをそれぞれ1行ずつ
//     順に記載し、4行目に送るメッセージを記載。
//  (例)
//    &D/)%X/C04S+#T'?+USM
//    *%/O-J_F$##5)CK`(%
//    #VGN-Z_*-FCG+VK1/E;M<47E1UCJ.C/`-U(S#F?C06CV)%C_0$$E8W$89(0[5(U
//    ボタンが押されました。
//---------------------------------------------------------------
#include <WiFi.h>
#include <WiFiClientSecure.h>
#include <SD.h>
#define SD_CS 16 // SD card chip select
#define SW    25
#define LED_G 33
#define LED_R 32
#define ON  1
#define OFF 0
#define FILENAME "/setup.txt"
// ssid,pw,tokenを暗号化したときのキー
char* crypt_key = "Open Sesame";
// LINE Notify
const char* host  = "notify-api.line.me";
char ssid[64],password[64],token[64],message[128];
int sw_newest;
void setup()
{
    Serial.begin(115200);
    pinMode(SW,INPUT_PULLUP);
    pinMode(LED_G,OUTPUT);
    pinMode(LED_R,OUTPUT);
    led_green(OFF);
    led_red(OFF);
    sw_newest = digitalRead(SW);
    if (read_setup_file(FILENAME)) {
      blink_red(10);
      led_red(ON);
      sleep;
    }
}
void loop(){
  if (sw_FallingEdge()) {
    wifi_connect();      // WiFi接続
    send_msg2line();     // LINE notifyにメッセージを送る
    wifi_disconnect();   // WiFi接続解除
    while(!digitalRead(SW));
    delay(100);          // チャタリング対策
  }  
}
int sw_FallingEdge() {
  int prev = sw_newest;
  sw_newest = digitalRead(SW);
  return prev & ~sw_newest;  // change 1 -> 0
}
void led_green(int onoff) {
  if (onoff==ON) digitalWrite(LED_G,LOW);
  else           digitalWrite(LED_G,HIGH);
}
void led_red(int onoff) {
  if (onoff==ON) digitalWrite(LED_R,LOW);
  else           digitalWrite(LED_R,HIGH);
}
void wifi_connect() {
    WiFi.begin(ssid, password);
    int onoff = 0;
    while (WiFi.status() != WL_CONNECTED) {
        delay(500);
        if (onoff==0) led_green(ON);
        else          led_green(OFF);
        onoff ^= 1;
    }
    // 以下5行は、普通は不要なハズ
    IPAddress gateway(192,168,0,1);
    IPAddress subnet(255,255,255,0);
    IPAddress dns1(8, 8, 8, 8); // Google DNS primery server
    IPAddress dns2(8, 8, 4, 4); // Google DNS secondary server
    WiFi.config(WiFi.localIP(),gateway,subnet,dns1,dns2);    
    led_green(ON);
}
void wifi_disconnect() {
    WiFi.disconnect(true);
    led_green(OFF);
}
void send_msg2line() {
  WiFiClientSecure client;
  led_red(ON);
  client.setInsecure();
  if (!client.connect(host, 443)) {
    blink_red(10);
    led_red(OFF);
    return;
  }
  
  String query = String("message=") + String(message);
  String request = String("") +
               "POST /api/notify HTTP/1.1\r\n" +
               "Host: " + host + "\r\n" +
               "Authorization: Bearer " + token + "\r\n" +
               "Content-Length: " + String(query.length()) +  "\r\n" + 
               "Content-Type: application/x-www-form-urlencoded\r\n\r\n" +
                query + "\r\n";
  client.print(request);
 
  while (client.connected()) {
    String line = client.readStringUntil('\n');
    if (line == "\r") {
      break;
    }
  }
  
  String line = client.readStringUntil('\n');
  led_red(OFF);
}
void blink_red(int no) {
  int onoff = 1;
  int i;
  for (i=0; i<no*2 ;i++) {
    delay(100);
    if (onoff==0) led_red(ON);
    else          led_red(OFF);
    onoff ^= 1;
  }
}
int read_setup_file(char* fileName) {
  char  buff[128];
  if (!SD.begin(SD_CS, SPI, 24000000)) return 1;
  File setupFile = SD.open(fileName,FILE_READ);
  if (setupFile) {
    readLine(setupFile,buff);
    decrypt(ssid,buff,crypt_key);
    
    readLine(setupFile,buff);
    decrypt(password,buff,crypt_key);
    
    readLine(setupFile,buff);
    decrypt(token,buff,crypt_key);
    
    readLine(setupFile,message);
    setupFile.close();
  }
  return 0;
}
// microSDから1行読み取る
int readLine(File fp, char *buff){
  int i = 0;
  if(!fp.available()) {
    return -1;
  }
  while (fp.available()) {
    char c = fp.read();
    if (c=='\r');
    else if (c=='\n') break;
    else buff[i++] = c;
  }
  buff[i] = 0;
  
  return i;
}
void decrypt( char*a, char*b, char* key) {
  char x[128];
  int i,j,k,m;
  // ascii文字をbinaryに戻す
  for(i=0; b[i]!=0; i++) {
    x[i] = b[i] - (int)'#';
  }
  m = i - 1;
  // 6bit -> 8bit
  j = 0;
  for(i=0; j<m ; i++) {
    switch (i%3) {
      case 0:
        a[i] = (x[j]&0x3F)<<2 | (x[j+1]&0x30)>>4;
        j++;
        break;
      case 1:
        a[i] = (x[j]&0x0F)<<4 | (x[j+1]&0x3C)>>2; 
        j++;
        break;
      case 2:
        a[i] = (x[j]&0x03)<<6 | x[j+1]&0x3F;
        j += 2;
        break;
    }
  }
  m = i;
  
  k = 0;
  int n = strlen(key);
  for (i=0; i<m; i++) {
    a[i] ^= key[k++];
    k %= n;
  }
  a[i]= 0;
}
//---------------------------------------------------------------
// EOF
//---------------------------------------------------------------