ESP32 WEB SERVER
2025.08.09

今回はマイコンボードESP32を使って小型WEBサーバを作ってみました。
画像を取扱う都合上、PSRAMを実装しているESP32-S2を使用しています。
コンテンツはmicroSDカードに保存してあります。
右側にあるQWIIC拡張アダプタは関係ありません。
●ファイル構成(検証用)
ファイル構成はApache WEBサーバなどでの一般的な配置にしています。
microSDカードのルートにhtmlディレクトリを作り、
それをWEBサーバのドキュメントルート(www-root)にしています
<www-root>
┣<css>
┃ ┗ style.css
┣<images>
┃ ┗ flower.jpg
┣ example.html
┗ favicon.ico
ESP32 WEBサーバのローカルIPアドレスは、192.168.11.66に設定しています
ここからはHTTPプロトコルに基づいた処理の流れの解説です。

パソコンなどでブラウザを起動して、コンテンツにアクセスします
http://192.168.11.66/example.html
ESP32側で受信したブラウザからのリクエストは下記のようになります
GET /example.html HTTP/1.1
Host: 192.168.11.66
Connection: keep-alive
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/138.0.0.0 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7
Accept-Encoding: gzip, deflate
Accept-Language: en-US,en;q=0.9,ja;q=0.8
ブラウザからのリクエスト取得方法は色々ありますが、HTTPプロトコルを把握したい場合は、client.read()が便利です
unsigned char *pt;
pt = inbuf;
while(1) {
if(client80.available()) {
*pt = client80.read();
pt++;
} else {
break;
}
}
*pt = 0x0;
for (pt=inbuf;*pt;pt++) Serial1.write(*pt);
GET に続くファイル名に基づいて、microSDカードからHTMLファイル(example.html)を読込み、
レスポンスヘッダーに続けて出力します。ヘッダーの改行コードは \r\n です。
HTTP/1.1 200 OK
Content-Length: 911
Content-Type: text/html
Accept-Charset: UTF-8
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html><head>
<title>ゆっくりラズパイ活用講座</title>
<meta name="keywords" content="ゆっくりラズパイ活用講座">
<meta name="description" content="ESP32 WEB SERVER">
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<meta http-equiv="Content-Style-Type" content="text/css">
<link href="/css/style.css" rel="stylesheet" type="text/css">
<meta http-equiv="Pragma" content="no-cache">
<meta http-equiv="Cache-Control" content="no-cache">
<link href="/favicon.ico" rel="shortcuticon" type="image/x-icon">
</head>
<body onload="displayImage()">
<div id="imageContainer"></div>
<script>
function displayImage() {
const img = document.createElement('img');
img.src = '/images/flower.jpg';
img.alt = 'FLOWER';
document.getElementById('imageContainer').appendChild(img);
}
</script>
Hello world
</body>
</html>
ブラウザ側がこのHTMLファイルを初めて読み込んだ際には、サーバに対してファビコンの送信を要求される場合があります。
ファビコン(favicon)とは、ウェブサイトのブラウザタブやブックマークなどに表示される小さなアイコンのことです。
GET /favicon.ico HTTP/1.1
Host: 192.168.11.66
Connection: keep-alive
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/138.0.0.0 Safari/537.36
Accept: image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8
Referer: http://192.168.11.66/example_01.html
Accept-Encoding: gzip, deflate
Accept-Language: en-US,en;q=0.9,ja;q=0.8
この場合は、レスポンスヘッダーに続けてファビコン画像を出力します
HTTP/1.1 200 OK
Content-Length: 906
Content-Type: image/x-icon
┏━━━━━━━━━━━━━━┓
┃ FAVICON バイナリ・データ ┃
┗━━━━━━━━━━━━━━┛
次に、スタイルシートを要求されます
GET /css/style.css HTTP/1.1
Host: 192.168.11.66
Connection: keep-alive
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/138.0.0.0 Safari/537.36
Accept: text/css,*/*;q=0.1
Referer: http://192.168.11.66/example_05.html
Accept-Encoding: gzip, deflate
Accept-Language: en-US,en;q=0.9,ja;q=0.8
スタイルシートを出力します
HTTP/1.1 200 OK
Content-Length: 1511
Content-Type: image/css
┏━━━━━━━━━━━━━━━━━━┓
┃ スタイルシート テキスト・データ ┃
┗━━━━━━━━━━━━━━━━━━┛
画像を要求されます
GET /images/flower.jpg HTTP/1.1
Host: 192.168.11.66
Connection: keep-alive
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/138.0.0.0 Safari/537.36
Accept: image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8
Referer: http://192.168.11.66/example_05.html
Accept-Encoding: gzip, deflate
Accept-Language: en-US,en;q=0.9,ja;q=0.8
画像データを出力します
HTTP/1.1 200 OK
Content-Length: 19322
Content-Type: image/jpeg
┏━━━━━━━━━━━━━┓
┃ 画像 バイナリ・データ ┃
┗━━━━━━━━━━━━━┛
画像などのサイズの大きい素材をブラウザ側に送信するときに、
SDカードから読込んだ画像ファイルを1バイトずつ送信していると時間が掛かります。
そのため、画像情報をメモリ上に読込み、一括送信します
if (file.size()<=OUTPUT_BUF_SIZE) {
if (file.read(outbuf,file.size())==file.size()) {
if (client80.write(outbuf, file.size())==file.size()) result = true;
}
}
ブラウザ側では受信データをもとに画面に表示されていきます

補足:画像取得に関して
HTML内の画像取得部分を見てください。
BODYタグで示される内容が読み込まれた時点で、JavaScriptが呼び出され、サーバ側からの画像ファイル送信を要求しています。
<body onload="displayImage()">
<div id="imageContainer"></div>
<script>
function displayImage() {
const img = document.createElement('img');
img.src = '/images/flower.jpg';
img.alt = 'FLOWER';
document.getElementById('imageContainer').appendChild(img);
}
</script>
Hello world
</body>
この部分には、下記のような記述も考えられます
<body>
<img src="/images/flower.jpg" width="280" height="280">
Hello world
</body>
このように記述した場合、当方の環境では画像をうまく取得できないこともありました
サーバ側ソースコード
microSDカードモジュールとのSPI接続ピン番号、WIFI接続情報は、動作環境に合わせて変更してください。
また、完成したソースコードではないので、適時修正してください。
#include <WiFi.h>
#include <Wire.h>
#include <SPI.h>
#include <SD.h>
#define SECRET_SSID "************"
#define SECRET_PASS "************"
const uint8_t PRIMARY_DNS[4] = {192,168,11,1};
const uint8_t GATEWAY[4] = {192,168,11,1};
const uint8_t SUBNETMASK[4] = {255,255,255,0};
const uint8_t LOCAL_IP[4] = {192,168,11,66};
#define SPI_MISO MISO
#define SPI_MOSI MOSI
#define SPI_SCK SCK
#define SPI_SD_CS A3
SPIClass SDSPI(HSPI);
#define SDSPEED 40000000
#define HTTP_PORT (80)
WiFiServer server80(HTTP_PORT);
WiFiClient client80;
#define TYPE_HTML 0x01
#define TYPE_JPEG 0x02
#define TYPE_GIF 0x04
#define TYPE_PNG 0x08
#define TYPE_CSS 0x10
#define TYPE_ICO 0x20
#define TYPE_JS 0x40
#define INPUT_BUF_SIZE 4096
#define OUTPUT_BUF_SIZE 524288
#define FILE_NAME_SIZE 64
unsigned char inbuf[INPUT_BUF_SIZE];
unsigned char *outbuf;
char filename[FILE_NAME_SIZE];
boolean server_enable;
long elapsed;
void setup()
{
if (!WiFi.config(LOCAL_IP, GATEWAY, SUBNETMASK, PRIMARY_DNS)) while(1);
WiFi.mode(WIFI_STA);
WiFi.begin(SECRET_SSID, SECRET_PASS);
while (1) {
if (WiFi.status() == WL_CONNECTED) {
server80.begin();
SDSPI.begin(SPI_SCK, SPI_MISO, SPI_MOSI, -1);
pinMode(SPI_SD_CS, OUTPUT);
while(!SD.begin(SPI_SD_CS, SDSPI,SDSPEED)) delay(100);
break;
}
delay(500);
}
server_enable = true;
outbuf = (unsigned char*)malloc(OUTPUT_BUF_SIZE);
if (!outbuf) while(1);
}
boolean availableHTTP()
{
if(!client80) { // Client not exist
client80 = server80.available();
if(client80) {
if (!client80.connected()) return false;
}
} else if (!client80.connected()) {
client80.stop();
client80 = server80.available();
if(client80) {
if (!client80.connected()) return false;
}
}
return client80.available();
}
void notFound(unsigned char ctype)
{
client80.print("HTTP/1.1 404 Not Found\r\n");
if (ctype==TYPE_HTML) {
client80.print("Content-Type: text/html\r\n");
client80.print("Accept-Charset: UTF-8\r\n");
client80.print("\r\n");
client80.print("<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\">\r\n");
client80.print("<html><head>\r\n");
client80.print("<title>404 Not Found</title>\r\n");
client80.print("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\">\r\n");
client80.print("</head><body>該当するページはありません</body></html>\r\n");
} else {
client80.print("\r\n");
}
}
int getRequest(char *fname)
{
unsigned char ch;
unsigned char *pt;
int len = 0;
pt = inbuf;
while(1) {
if(client80.available()) {
*pt = client80.read();
pt++;
} else {
break;
}
}
*pt = 0x0;
pt = inbuf;
while (*pt) {
switch (*pt) {
case 'G':
if (memcmp(pt,"GET ",4)==0) {
pt+=4;
for (len=0; (*pt!=' ')&&(len<(FILE_NAME_SIZE-1));len++,pt++) *(fname+len) = *pt;
*(fname+len) = 0x0;
return len;
}
break;
}
while ((*pt!=0x0a)&&(*pt!=0x0)) pt++;
}
return len;
}
boolean putHTML(char *fname)
{
char line[64];
char path[64];
unsigned char *html_data = 0x0;
boolean result = false;
sprintf(path,"/html%s",fname);
File file = SD.open(path);
if (file) {
client80.print("HTTP/1.1 200 OK\r\n");
sprintf(line,"Content-Length: %d\r\n",file.size());
client80.print(line);
client80.print("Content-Type: text/html\r\n");
client80.print("Accept-Charset: UTF-8\r\n");
client80.print("\r\n");
if (file.size()<=OUTPUT_BUF_SIZE) {
if (file.read(outbuf,file.size())==file.size()) {
if (client80.write(outbuf, file.size())==file.size()) result = true;
}
}
}
client80.print("\r\n");
if (file) file.close();
return result;
}
boolean putImage(char *fname, unsigned char ctype)
{
char path[64];
char line[64];
unsigned char *img_data = 0x0;
boolean result = false;
sprintf(path,"/html%s",fname);
File file = SD.open(path);
if (file) {
client80.print("HTTP/1.1 200 OK\r\n");
sprintf(line,"Content-Length: %d\r\n",file.size());
client80.print(line);
switch (ctype) {
case TYPE_JPEG:
client80.print("Content-Type: image/jpeg\r\n\r\n");
break;
case TYPE_GIF:
client80.print("Content-Type: image/gif\r\n\r\n");
break;
case TYPE_PNG:
client80.print("Content-Type: image/png\r\n\r\n");
break;
case TYPE_CSS:
client80.print("Content-Type: text/css\r\n");
client80.print("Accept-Charset: UTF-8\r\n\r\n");
break;
case TYPE_ICO:
client80.print("Content-Type: image/x-icon\r\n\r\n");
break;
case TYPE_JS:
client80.print("Content-Type: text/javascript\r\n\r\n");
break;
}
if (file.size()<=OUTPUT_BUF_SIZE) {
if (file.read(outbuf,file.size())==file.size()) {
if (client80.write(outbuf, file.size())==file.size()) result = true;
}
}
}
if (file) file.close();
return result;
}
void loop()
{
int filesize;
if (availableHTTP()) {
filesize = getRequest((char*)&filename);
if (filesize > 0) {
char *pt = filename;
while ((*pt!='.')&&(*pt!=0x0)) pt++;
if (memcmp(pt,".html",5)==0) {
if (!putHTML(filename)) notFound(TYPE_HTML);
} else if (memcmp(pt,".css",4)==0) {
if (!putImage(filename, TYPE_CSS)) notFound(TYPE_CSS);
} else if (memcmp(pt,".jpg",4)==0) {
if (!putImage(filename, TYPE_JPEG)) notFound(TYPE_JPEG);
} else if (memcmp(pt,".gif",4)==0) {
if (!putImage(filename, TYPE_GIF)) notFound(TYPE_GIF);
} else if (memcmp(pt,".png",4)==0) {
if (!putImage(filename, TYPE_PNG)) notFound(TYPE_PNG);
} else if (memcmp(pt,".ico",4)==0) {
if (!putImage(filename, TYPE_ICO)) notFound(TYPE_ICO);
} else if (memcmp(pt,".js",3)==0) {
if (!putImage(filename, TYPE_JS )) notFound(TYPE_JS);
} else {
notFound(TYPE_HTML);
}
}
}
}
ネット環境により、ブラウザにコンテンツの一部が取り込まれない状況が発生します。
上記はサンプルプログラムで、ブラッシュアップが必要です。
参考程度にみてください。
応用するとこんな学習サイトを作ることも可能です

スクリプトによりサーバ側からJSON形式の教材データをダウンロードして使用しています。
メイン画面をタップすることで答えと解説が表示しています
|
Raspberry Pi(ラズベリー パイ)は、ARMプロセッサを搭載したシングルボードコンピュータ。イギリスのラズベリーパイ財団によって開発されている。
Arduinoで学ぶ組込みシステム入門(第2版)
●Arduinoを使って組込みシステム開発を理解する
・ハードウェアやソフトウェアなどの基礎知識/
・設計から実装までを系統的に説明するモデルベース開発/
・Arduinoを用いた実際の開発例
最新 使える! MATLAB 第3版
◆◆すぐに「使える!」 全ページフルカラー!◆◆
・MATLAB R2022bに対応し、解説もより詳しく!/
・コマンド・スクリプトの例が豊富で、動かして学べる!/
・超基本から解説。これから使いはじめる人にぴったり!/
・全編フルカラー、スクリーンショットも豊富!
Amazon Web Services基礎からのネットワーク&サーバー構築改訂4版
1.システム構築をインフラから始めるには/
2.ネットワークを構築する/
3.サーバーを構築する/
4.Webサーバーソフトをインストールする/
5.HTTPの動きを確認する/
6.プライベートサブネットを構築する/
7.NATを構築する/
8.DBを用いたブログシステムの構築/
9.TCP/IPによる通信の仕組みを理解する
C言語は第二の母国語: 独学学生時代から企業内IT職人時代に培った、独立のための技術とノウハウ 平田豊著
学生時代から独学でプログラミングをはじめ、企業内でデバイスドライバを開発し、そして独立後もたくさんのアプリケーション開発や技術書制作に携わってきた著者。その筆者が大事に使い続ける「C言語」の“昔と今”について、気づいたことや役立つ知識、使ってきたツールなどについて、これまで記してきたことを整理してまとめました。
本書では、現役プログラマーだけでなく、これからプログラミングを学ぶ学生などにも有益な情報やノウハウを、筆者の経験を元に紹介しています。
1冊ですべて身につくJavaScript入門講座
・最初の一歩が踏み出せる! 初心者に寄り添うやさしい解説
・最新の技術が身につく! 今のJavaScriptの書き方・使い方
・絶対に知っておきたい! アニメーションとイベントの知識
・プログラミングの基本から実装方法まですべて学べる
図解! Git & GitHubのツボとコツがゼッタイにわかる本
ソフトウェア開発では欠かすことのできないGit、GitHub。
これからGit、GitHubを使いたいという入門者の方でも、実際に手を動かしながら使い方を学べます。
C自作の鉄則!2023 (日経BPパソコンベストムック)
メーカー製のパソコンはスペックが中途半端で、自分が本当に欲しい機種がない――。そう思っている人には、ぜひ自作パソコンをお薦めします。自作パソコンのパーツは進化が速く、しかも驚くほど種類が豊富。価格も性能も、幅広く用意されているため、満足度100%の“自分だけの1台”を手に入れることができます。
Interface 2023年6月号
特集:第1部 フィルタ設計 基礎の基礎/
第2部 係数アプリや波形観測アプリで合点!FIR&IIRフィルタ作り/
第3部 配布プリント基板で体験!マイコンで動くフィルタ作り
日経Linux 2023年5月号
【特集 1】 AI時代の最強フリーソフト ~ 25のやりたいを実現!
【特集 2】 AWS、Azureのうまみを無料で体感!面倒なことはクラウドに任せよう
【特集 3】 新しいRaspberry Pi Cameraで遊んでみよう
【特集 4】 Linuxで旧型PCを復活! 1kg切るモバイルPCを「ChromeOS Flex」でChromebook化
ラズパイマガジン2022年秋号
特集:5大人気ボード 電子工作超入門
「半導体不足で在庫が不足し、電子工作のボードがなかなか買えない…」。そんな今にふさわしい特集を企画しました。5種の人気ボードにすべて対応した電子工作の入門特集です。「GPIO」や「I2C」を使った電子パーツの制御方法は、どのボードでも同じです。手に入れられたボードを使って、今こそ電子工作を始めましょう。
地方で稼ぐ! ITエンジニアのすすめ
学歴、理系の知識、専門スキル……全部なくてもITエンジニアになれる!
地方でも高収入でやりがいをもって働ける!ITエンジニアの魅力を一挙大公開
Raspberry Piのはじめ方2022
本書は、ラズパイやPicoの買い方やインストール、初期設定といった基本から、サーバー、電子工作、IoT、AIといったラズパイならではの活用方法まで、1冊でお届けします。
ラズパイをこれから始める方向けに、全36ページの入門マンガ「女子高生とラズベリーパイ」も巻末に掲載。これを読むだけでラズパイがどんなものなのか、すぐに分かって触れるようになります。
ハッカーの学校 IoTハッキングの教科書
生活にとけこみ、家電機器を便利にするIoT技術。
Webカメラなど、便利の裏側に潜むセキュリティの危険性をハッキングで検証。
専門家がパケットキャプチャからハードウェアハッキングまで、その攻撃と防御を徹底解説。
本書は2018年7月に刊行された「ハッカーの学校IoTハッキングの教科書」に一部修正を加えた第2版です。
攻撃手法を学んで防御せよ! 押さえておくべきIoTハッキング
本書は、経済産業省から2021年4月にリリースされた、IoTセキュリティを対象とした『機器のサイバーセキュリティ確保のためのセキュリティ検証の手引き』の『別冊2 機器メーカに向けた脅威分析及びセキュリティ検証の解説書』をもとに、IoT機器の開発者や品質保証の担当者が、攻撃者の視点に立ってセキュリティ検証を実践するための手法を、事例とともに詳細に解説しました。
ポチらせる文章術
販売サイト・ネット広告・メルマガ・ブログ・ホームページ・SNS…
全WEB媒体で効果バツグン!
カリスマコピーライターが教える「見てもらう」「買ってもらう」「共感してもらう」すべてに効くネット文章術
プログラマーは世界をどう見ているのか 西村博之著
イーロン・マスク(テスラ)、ジェフ・べゾス(Amazon)、ラリー・ペイジ(Google)…etc.
世界のトップはなぜプログラマーなのか?
ニーア オートマタ PLAY ARTS改 <ヨルハ 二号 B型 DX版> PVC製 塗装済み可動フィギュア
「NieR:Automata」より、ヨルハ二号B型こと2BがPLAY ARTS改に新たに登場!
高級感の感じられるコスチュームや髪の質感、洗練されたボディバランス、細かなデティールに至るまでこだわり抜かれた逸品。
DX版には通常版のラインナップに加え2Bの随行支援ユニット ポッド042などをはじめ“純白の美しい太刀"白の約定やエフェクトパーツ、自爆モードを再現できる換装用ボディパーツ、シーンに合わせて変えられる顔パーツ2種も付属する豪華な仕様に。
作中のあらゆるシーンを再現することが可能なファン必見の一品となっている。
Newtonライト2.0 ベイズ統計
ベイズ統計は,結果から原因を推定する統計学です。AIや医療などの幅広い分野で応用されています。その基礎となるのは18世紀に考えだされた「ベイズの定理」です。
この本では,ベイズ統計学のきほんをやさしく紹介していきます。
白光(HAKKO) ダイヤル式温度制御はんだ吸取器 ハンディタイプ FR301-81
無水エタノールP 500mlx2個パック(掃除)
ケイバ(KEIBA) マイクロニッパー MN-A04
サンハヤト SAD-101 ニューブレッドボード
白光(HAKKO) HEXSOL 巻はんだ 精密プリント基板用 150g FS402-02
[Amazon限定ブランド]【指定第2類医薬品】PHARMA CHOICE 解熱鎮痛薬 解熱鎮痛錠IP 100錠
|