ESP32でWake ON LAN!外出先からPCを起動してみよう!

もろもろ

ESP32を使って、外出先からパソコンの電源を入れられるようにしてみました。

GitHub - takabus/esp32wol: Wake on LAN Repeater with ESP32/ESP8266
Wake on LAN Repeater with ESP32/ESP8266. Contribute to takabus/esp32wol development by creating an account on GitHub.
スポンサーリンク

WakeOnLANとは?

WakeOnLANとは、ネットワークにつながっているパソコンを遠隔操作で起動できる技術のことをいいます。

起動したいMACアドレスを指定し、マジックパケットという特殊なパケットを送信することにより、任意のコンピューターを起動させることができます。

なお、起動したいコンピューターについては、あらかじめWakeOnLANを有効に設定しておく必要があります。BIOSの設定が必要になることがありますので、メーカーのサイトなどで確認してください。

スポンサーリンク

まずは完成形(^^)!

最終的にはこのような感じになりました。

EPS32wolのWebダッシュボードのスクリーンショット

ブラウザでhttp://[ESP32のIPアドレス]/webにアクセスすると、このようなページが表示されます。

MACアドレスを入力し、起動ボタンを押すと、ESP32からマジックパケットが送信され、そのMACアドレスのパソコンを起動させることができます。

遊び心で、タイトルをカラフルにしてみました(^^)

全体のソースコードはGitHubに置いておきますね。

GitHub - takabus/esp32wol: Wake on LAN Repeater with ESP32/ESP8266
Wake on LAN Repeater with ESP32/ESP8266. Contribute to takabus/esp32wol development by creating an account on GitHub.

どういう仕組み?

処理の流れを解説していきます。

① ブラウザからアクセス

まず、ブラウザでhttp://[ESP32のIPアドレス]/wolにアクセスすると、ESP32がSPIFFSに保存されているhtmlファイルを返します。

 SPIFFSというのは、ESP32が持っているファイルシステムのことです。

 ESP32は、SPIFFSというUSBメモリのようなストレージを持っています。ここに任意のファイルを保存しておき、ブラウザからアクセスされたときにファイルの内容を返すようにすることができます。

 SPIFFSを活用すると、HTMLをソースコードに埋め込む必要がなくなるので、大変便利です。ESP32にWebアプリを組み込みたいときは、ぜひ使用することをおすすめします。

 

 Webサーバーには、ESPAsyncWebServerを採用しました。最大の特長は「Async」つまり非同期で動作することです。

GitHub - me-no-dev/ESPAsyncWebServer: Async Web Server for ESP8266 and ESP32
Async Web Server for ESP8266 and ESP32. Contribute to me-no-dev/ESPAsyncWebServer development by creating an account on GitHub.

 ESP32で動作するWebサーバーライブラリにはさまざまなものがあります。有名どころでは、その名の通り「WebServer」などがありますが、これらのライブラリの多くは同期処理となっています。

 同期処理のWebサーバーの場合、loop内でアクセスを待機する必要があります。アクセスがあると、ループが回って、クライアントにレスポンスを返し、処理が終了したらループを中断して、次のアクセスを待つという処理形態をとっています。そのため、定期的に実行したい処理が別にあった場合には、Webサーバーの動作を止めるなどの工夫が必要となります。

これだと、一定時間ごとに行いたい処理を追加することができません。

 一方で、ESPAsyncWebServerは非同期処理に対応しているので、loopで明示的にアクセスを待機する必要がありません。

 このため、ESPAsyncWebServerを使用すると、定期的に実行する処理とWebサーバーを共存させることができます。

 将来的に同じESP32でWOL以外の処理も動かしたくなったときに便利なので、ESPAsyncWebServerを採用しました。

自分の場合は、今回作成したWOLリピーターに温湿度センサーを追加しています。定期的にサーバーへセンサーデータをPOSTするようにし、環境データを蓄積するのに使用しています。

Webサーバーが非同期で動作してくれると、あとからでも、こういったループ処理をかんたんに組み込むことができます。

② 「起動ボタン」をタップする

 MACアドレスを入力し、「起動」ボタンを押すと、axiosによりhttp://[ESP32のIPアドレス]/api/wolにMACアドレスがPOSTされます。

 これを受けて、ESP32はマジックパケットを送信し、処理の結果が表示されます。マジックパケットの送信はWakeOnLanというライブラリを採用しました。

GitHub - a7md0/WakeOnLan: Wake On LAN Library (ESP8266 & ESP32)
Wake On LAN Library (ESP8266 & ESP32). Contribute to a7md0/WakeOnLan development by creating an account on GitHub.

 フロントエンドについては、CSSフレームワークとしてBootstrap、JavascriptフレームワークとしてVue.jsを使用しています。C/C++での文字列処理は大変ですので、ESP32はもっぱらAPIサーバーとしておき、フロントエンドの処理はすべてVue.jsで行うようにしています。

つくってみよう!

それではさっそく作り方をご紹介しましょう。

GitHubのソースコードはこちらになります。せっかちな方はクローンしちゃえば、一発でビルドできますよ。

GitHub - takabus/esp32wol: Wake on LAN Repeater with ESP32/ESP8266
Wake on LAN Repeater with ESP32/ESP8266. Contribute to takabus/esp32wol development by creating an account on GitHub.

そうでない方・Gitに慣れていない方は、下記の手順でつくってみてください。

【手順1】PlatformIOでプロジェクトを作成

Visual Studio CodeにPlatformIOをインストールします。

Homeページの[New Project]をクリックして、プロジェクトを作成します。

プロジェクト名を設定します。

ボードとフレームワークを選択します。

ボードは「Espressif ESP32 Dev Module」、フレームワークは「Arduino」を選択します。

[Finish]ボタンを押して、いざ作成!

【手順2】ライブラリをインストール

マジックパケットの送信は「WakeOnLan」ライブラリを使用します。

GitHub - a7md0/WakeOnLan: Wake On LAN Library (ESP8266 & ESP32)
Wake On LAN Library (ESP8266 & ESP32). Contribute to a7md0/WakeOnLan development by creating an account on GitHub.

WOL.sendMagicPacket(MACアドレス)の1文だけで、マジックパケットを送信することができます。必要に応じて、ポート番号やパスワードの指定も可能です。

PlatformIOではコマンドラインからライブラリをインストールすることができます。

ライブラリのページにアクセスし、インストールするためのコマンドをコピーしてください。

a7md0/WakeOnLan: Generate and send Wake On Lan (WOL) packet over UDP protocol.
Generate and send Wake On Lan (WOL) packet over UDP protocol.

PlatformIOでコンソールを起動し、

コピーしたコマンドを貼り付けて、実行します。

pio lib install "a7md0/WakeOnLan"

ESP32をWebサーバーにするためのライブラリ「ESPAsyncWebServer」もインストールしておきましょう。

GitHub - esphome/ESPAsyncWebServer: Async Web Server for ESP8266 and ESP32
Async Web Server for ESP8266 and ESP32. Contribute to esphome/ESPAsyncWebServer development by creating an account on GitHub.

同様にライブラリページからコマンドをコピーして、PlatformIOのコンソールにペーストしインストールします。

ottowinter/ESPAsyncWebServer-esphome: Asynchronous HTTP and WebSocket Server Library for ESP8266 and ESP32
Asynchronous HTTP and WebSocket Server Library for ESP8266 and ESP32

【手順3】プログラミングしよう

src/main.cppを編集して、プログラムを書いていきましょう!

ソースコードはこんな感じです▼(最新版はGitHubを参照してくださ~い)

#include <Arduino.h>
#include <WiFi.h>
#include <WiFiUDP.h>
#include <WakeOnLan.h>
#include "ESPAsyncWebServer.h"
#include "SPIFFS.h"

const char *ssid = "SSID";
const char *password = "KEY";
AsyncWebServer server(80);
WiFiUDP UDP;
WakeOnLan WOL(UDP);

void logWebServer();

void setup()
{
  // put your setup code here, to run once:

  //
  // シリアル通信の有効化
  //
  Serial.begin(115200);
  delay(5000);
  Serial.println();

  //
  // SPIFFS(ファイルシステム)の有効化
  //
  SPIFFS.begin();

  //
  // Wi-Fiに接続
  //

  Serial.print("Connecting to ");
  Serial.println(ssid);
  WiFi.begin(ssid, password);
  while (WiFi.status() != WL_CONNECTED)
  {
    delay(500);
    Serial.print(".");
  }
  Serial.println();
  Serial.println("WiFi connected.");
  Serial.print("IP address: ");
  Serial.println(WiFi.localIP());

  //
  // WebServerの実装
  //

  server.on("/api/wol", HTTP_ANY, [](AsyncWebServerRequest *server) {
    logWebServer();
    String target = "";

    if (server->method() == HTTP_GET)
    {
      // レスポンスを返す
      server->send(200, "application/json", "{\"status\":\"ready\"}");
    }

    if (server->method() == HTTP_POST)
    {
      Serial.print("[WOL]");

      // MACアドレスを取得
      target=server->arg("mac");
      Serial.print(target);

      // WOLを実行
      WOL.sendMagicPacket(target.c_str());
      Serial.println("...done");
      // レスポンスを返す
      server->send(200, "application/json", "{\"status\":\"ok\",\"target\":\"" + target + "\"}");
    }
  });

  server.on("/wol", HTTP_GET, [](AsyncWebServerRequest *server) {
    logWebServer();
    File file = SPIFFS.open("/wol.html", "r");
    server->send(200, "text/html", file.readString());
    file.close();
  });

  server.onNotFound([](AsyncWebServerRequest *server) {
    server->send(404, "text/html", "<h1>Not Found</h1>");
  });

  // CORSを有効化する
  DefaultHeaders::Instance().addHeader("Access-Control-Allow-Origin", "*");
  DefaultHeaders::Instance().addHeader("Access-Control-Allow-Headers", "*");

  // WebServerを開始する
  server.begin();
  Serial.println("Server started.");

  //
  // セットアップ完了!
  //
  Serial.println("Ready!");
}

void loop()
{
  // put your main code here, to run repeatedly:
}

// Webサーバーへのアクセスをコンソールに出力する関数
void logWebServer()
{
  Serial.print("[HTTP][");
  // switch (server->method())
  // {
  // case HTTP_GET:
  //   Serial.print("GET");
  //   break;
  // case HTTP_POST:
  //   Serial.print("POST");
  // default:
  //   break;
  // }
  Serial.print("]");
  // Serial.println(server.uri());
}

SSIDとKEYのところは、ご自分のWiFi環境にあわせてください。

なお、ESP32は802.11acには対応していないので注意!802.11nのSSIDを指定しましょう。

フロントエンドを作成しよう

つづけて、ブラウザで表示されるWebページを作成しましょう。

WebページはESP32のSPIFFS内に保存するようにします。

プロジェクト直下にdataフォルダを作成し、ファイルを配置することで、SPIFFSにファイルを転送することができます。

dataフォルダを作成したら、htmlファイル(wol.html)を作成します。

<!doctype html>
<html lang="ja">

<head>
    <!-- Required meta tags -->
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
    <!-- Bootstrap CSS -->
    <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.0.0-beta1/dist/css/bootstrap.min.css" rel="stylesheet"
        integrity="sha384-giJF6kkoqNQ00vy+HMDP7azOuL0xtbfIcaT9wjKHr8RbDVddVHyTfAAsrekwKmP1" crossorigin="anonymous">
    <title>ホーム|ESP32WOL</title>

    <style>
        [v-cloak] {
            display: none;
        }

        .fuwafuwa {
            animation: fuwafuwa 6s infinite ease-in-out .6s alternate;
            display: inline-block;
            transition: 1.5s ease-in-out;
            width: 70px;
            margin-top: 15px;
        }

        @keyframes fuwafuwa {
            0% {
                transform: translate(0, 0);
            }

            50% {
                transform: translate(0, -5px);
            }

            100% {
                transform: translate(0, 0);
            }
        }
    </style>
</head>

<body>
    <div id="app" v-cloak>
        <div class="container" style="max-width: 1000px;">
            <h3 class="mt-3">
                <img class="fuwafuwa"
                    src=""
                    height="50px">
                <span style="color:#1fd857">ESP32</span> <span style="color:#ff9900">WakeOnLAN</span></h3>
            <hr>
            <div v-show="alert.type" :class="[alert.type]" class="alert" role="alert">
                {{alert.msg}}
            </div>

            <div class="form-group">
                <h3>MACアドレスを指定して起動</h3>
                <input v-model="selectItem.mac" type="text" class="form-control" name="" id="mac"
                    aria-describedby="helpId" placeholder="XX:XX:XX:XX:XX:XX">
                <p id="helpId" class="form-text text-muted">MACアドレスを指定して、対象のコンピューターを起動します。 </p>
                <button @click="wolAlert" class="btn btn-secondary w-100" :class="{'btn-danger':validMac}"
                    :disabled="!validMac">マジックパケットを送信する</button>

            </div>
            <div class="form-group mt-3">
                <h3>リストから選んで起動</h3>
                <p id="listhelp" class="form-text text-muted">クリックしたマシンを起動します。</p>

                <table class="table table-hover" aria-describedby="listhelp">
                    <thead>
                        <tr>
                            <th>ホスト名</th>
                            <th>MACアドレス</th>
                        </tr>
                    </thead>
                    <tbody>
                        <tr v-for="item in maclist" @click="selectMac(item)" style="cursor:pointer">
                            <td scope="row">{{item.name}}</td>
                            <td>{{item.mac}}</td>
                        </tr>
                    </tbody>
                </table>
            </div>
        </div>

        <div class="modal " tabindex="-1" id="myModal">
            <div class="modal-dialog  modal-dialog-centered">
                <div class="modal-content">
                    <div class="modal-header">
                        <h5 class="modal-title">確認</h5>
                        <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
                    </div>
                    <div class="modal-body">
                        <p>起動してよろしいですか?</p>
                        <p><b>ホスト名:{{selectItem.name}}</b></p>
                        <p><b>MACアドレス:{{selectItem.mac}}</b></p>
                    </div>
                    <div class="modal-footer">
                        <button type="button" class="btn btn-lg btn-danger w-100" @click="sendMac">起動する</button>
                        <button type="button" class="btn btn-secondary w-100" data-bs-dismiss="modal">キャンセル</button>
                    </div>
                </div>
            </div>
        </div>
    </div>
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
    <!-- JavaScript Bundle with Popper -->
    <script src="https://cdn.jsdelivr.net/npm/bootstrap@5.0.0-beta1/dist/js/bootstrap.bundle.min.js"
        integrity="sha384-ygbV9kiqUc6oa4msXn9868pTtWMgiQaeYH7/t7LECLbyPA2x65Kgf80OJFdroafW"
        crossorigin="anonymous"></script>
    <script>
        var app = new Vue({
            el: "#app",
            data: {
                selectItem: { name: "", mac: "" },
                maclist: [
                    { name: "host1", mac: "xx:xx:xx:xx:xx:xx" },
                    { name: "host2", mac: "xx:xx:xx:xx:xx:xx" },
                    { name: "host3", mac: "xx:xx:xx:xx:xx:xx" },

                ],
                alert: {
                    type: null,
                    msg: null,
                },
                // モーダル
                myModal,
            },
            mounted() {
                // モーダルウィンドウを初期化する
                this.myModal = new bootstrap.Modal(document.getElementById('myModal'), null);
            },
            computed: {
                // MACアドレスが有効かバリデーションする
                validMac: function () {
                    // MACアドレスの正規表現を指定する
                    let filterStr = /^([0-9A-F]{2}[:-]){5}([0-9A-F]{2})$/i;
                    // チェックして、一致していればtrueを返す
                    return filterStr.test(this.selectItem.mac);
                }
            },
            methods: {
                // WOL実行確認ダイアログを表示する
                wolAlert() {
                    // モーダルを表示する
                    this.myModal.show();
                },
                // WOLを実行する
                sendMac: function () {

                    // POSTデータを作成する

                    var formdata = new FormData();
                    formdata.append("mac", this.selectItem.mac);

                    // POSTする

                    axios.post("/api/wol", formdata).then((res) => {
                        if (res.data.status == "ok") {
                            this.alert.type = "alert-success";
                            this.alert.msg = "実行しました!";
                        } else {
                            this.alert.type = "alert-danger";
                            this.alert.msg = "エラーが発生しました";
                        }
                    }).catch((err) => {
                        this.alert.type = "alert-danger";
                        this.alert.msg = "エラーが発生しました[例外]";
                    }).finally(() => {
                        this.myModal.hide();
                    });
                },
                // リストから起動するマシンを選択する
                selectMac(item) {
                    this.selectItem = JSON.parse(JSON.stringify(item));
                    this.wolAlert();
                }
            }
        })
    </script>
</body>

</html>

ブラウザに実行結果などを表示したい場合、ネット上で探してみると、ESP32側で文字列処理してHTMLを生成している方が多いようです。

しかし、ここはフレームワークを活用することを強くおすすめします。

とくにおすすめなのが、Vue.jsaxiosの組み合わせ。axiosをつかって、ESP32との通信はもっぱらjsonで行うようにし、表示はVue.jsにまかせるようにします。すると、ESP32の開発も、フロントエンドの開発も、非常にかんたんになります。

「Vue.jsってなに?」という方は、こちらのページをご覧いただくとイメージがつかめると思います。要するに複雑な文字列の処理からおさらばできるようになります。

はじめに — Vue.js
Vue.js - The Progressive JavaScript Framework

ビルドしよう

いよいよビルドしてみましょう。

ビルドとは?

ソースコードを機械が読み込める形に変換することをビルドと呼びます。

Pythonなどのようにソースコードのまま実行することができる言語(インタプリタ型言語)もありますが、C++はビルドしないと実行することができません。

チェックマークをクリックして、プログラムをビルドします。

Successが表示されれば、ビルド成功です。

つづけて、ビルドしたファームウェアをESP32にアップロードしましょう。ESP32をUSBで接続し、をクリックします。

Successが表示されたらOKです。

ESP32にファイルをアップロードしよう

dataフォルダ内にあるhtmlファイルもアップロードしましょう。

PlatformIOでコンソールを開き、次のコマンドを実行します。

pio run --target uploadfs

バイナリのアップロードとは別にアップロードできるのもSPIFFSのメリットです。

HTMLファイルのみ変更した場合は、リビルドする必要はありません。ファイルのアップロードのみサクッと行うことができます。

ファイルのアップロードに成功すると、ESP32が再起動します。

再起動が完了したら、ブラウザからhttp://[ESP32のIPアドレス]/wolにアクセスしてみましょう。きちんと動作するはずです。

ESP32のIPアドレスを調べるときは、Network Analyzerというアプリがおすすめです。

https://apps.apple.com/jp/app/network-analyzer/id562315041
Network Analyzer - Google Play のアプリ
無線LANスキャナ、シグナルメーターは、ping、tracerouteを、WHOIS、DNSクエリ&他のネットツール

[SCAN]ボタンをタップするだけで、ササッとIPアドレスを確認できます。

Pingやnslookupにも対応しているので、いろいろ使えますよ~

外出先から起動できるようにするには?

ここまでで、ローカルネットワーク内からはマジックパケットを送れるようになりました。では、外出先からESP32にアクセスするには、どうすればいいのでしょうか。

これはルーターのポート開放で実現できます。

市販のルーターの設定ページにアクセスすると、「静的マスカレード」または「ポート開放」といった設定項目があるはずです。これを設定して、外部のポートをESP32の80/TCPポートに転送するように設定してください。これで、ブラウザから自宅のグローバルIPへアクセスすると、インターネットからESP32にアクセスできるようになります。

なお、外部公開する場合は、セキュリティに配慮して運用するようにしましょう。

今回使用しているESPAsyncWebServerはBASIC認証も実装することができます。外部公開する前に、ソースコードをカスタマイズして、実装しておきましょう!

まとめ

ESP32でWakeOnLanする方法を紹介しました。

安価で入手できるESP32ですが、使い方は多岐にわたります。さまざまなライブラリが公開されているので、いろんな使い方を模索してみてはいかがでしょうか。

なお、PCをWOLで起動するには、設定の変更が必要となることがありますので、パソコンのサポートページなどで確認してみてください。

↓改造して、ボタンからも起動できるようにしてみました!

コメント

タイトルとURLをコピーしました