Youen
7 months ago
57 changed files with 314 additions and 9871 deletions
@ -1,36 +0,0 @@ |
|||||||
#!/bin/bash |
|
||||||
|
|
||||||
export ARDUINO_ESP32_PATH="$ARDUINO_USR_PATH/hardware/espressif/esp32" |
|
||||||
if [ ! -d "$ARDUINO_ESP32_PATH" ]; then |
|
||||||
echo "Installing ESP32 Arduino Core ..." |
|
||||||
script_init_path="$PWD" |
|
||||||
mkdir -p "$ARDUINO_USR_PATH/hardware/espressif" |
|
||||||
cd "$ARDUINO_USR_PATH/hardware/espressif" |
|
||||||
|
|
||||||
echo "Installing Python Serial ..." |
|
||||||
pip install pyserial > /dev/null |
|
||||||
|
|
||||||
if [ "$OS_IS_WINDOWS" == "1" ]; then |
|
||||||
echo "Installing Python Requests ..." |
|
||||||
pip install requests > /dev/null |
|
||||||
fi |
|
||||||
|
|
||||||
if [ "$GITHUB_REPOSITORY" == "espressif/arduino-esp32" ]; then |
|
||||||
echo "Linking Core..." |
|
||||||
ln -s $GITHUB_WORKSPACE esp32 |
|
||||||
else |
|
||||||
echo "Cloning Core Repository..." |
|
||||||
git clone https://github.com/espressif/arduino-esp32.git esp32 > /dev/null 2>&1 |
|
||||||
fi |
|
||||||
|
|
||||||
echo "Updating Submodules ..." |
|
||||||
cd esp32 |
|
||||||
git submodule update --init --recursive > /dev/null 2>&1 |
|
||||||
|
|
||||||
echo "Installing Platform Tools ..." |
|
||||||
cd tools && python get.py |
|
||||||
cd $script_init_path |
|
||||||
|
|
||||||
echo "ESP32 Arduino has been installed in '$ARDUINO_ESP32_PATH'" |
|
||||||
echo "" |
|
||||||
fi |
|
@ -1,29 +0,0 @@ |
|||||||
#!/bin/bash |
|
||||||
|
|
||||||
echo "Installing ESP8266 Arduino Core ..." |
|
||||||
script_init_path="$PWD" |
|
||||||
mkdir -p "$ARDUINO_USR_PATH/hardware/esp8266com" |
|
||||||
cd "$ARDUINO_USR_PATH/hardware/esp8266com" |
|
||||||
|
|
||||||
echo "Installing Python Serial ..." |
|
||||||
pip install pyserial > /dev/null |
|
||||||
|
|
||||||
if [ "$OS_IS_WINDOWS" == "1" ]; then |
|
||||||
echo "Installing Python Requests ..." |
|
||||||
pip install requests > /dev/null |
|
||||||
fi |
|
||||||
|
|
||||||
echo "Cloning Core Repository ..." |
|
||||||
git clone https://github.com/esp8266/Arduino.git esp8266 > /dev/null 2>&1 |
|
||||||
|
|
||||||
echo "Updating submodules ..." |
|
||||||
cd esp8266 |
|
||||||
git submodule update --init --recursive > /dev/null 2>&1 |
|
||||||
|
|
||||||
echo "Installing Platform Tools ..." |
|
||||||
cd tools |
|
||||||
python get.py > /dev/null |
|
||||||
cd $script_init_path |
|
||||||
|
|
||||||
echo "ESP8266 Arduino has been installed in '$ARDUINO_USR_PATH/hardware/esp8266com'" |
|
||||||
echo "" |
|
@ -1,228 +0,0 @@ |
|||||||
#!/bin/bash |
|
||||||
|
|
||||||
#OSTYPE: 'linux-gnu', ARCH: 'x86_64' => linux64 |
|
||||||
#OSTYPE: 'msys', ARCH: 'x86_64' => win32 |
|
||||||
#OSTYPE: 'darwin18', ARCH: 'i386' => macos |
|
||||||
|
|
||||||
OSBITS=`arch` |
|
||||||
if [[ "$OSTYPE" == "linux"* ]]; then |
|
||||||
export OS_IS_LINUX="1" |
|
||||||
ARCHIVE_FORMAT="tar.xz" |
|
||||||
if [[ "$OSBITS" == "i686" ]]; then |
|
||||||
OS_NAME="linux32" |
|
||||||
elif [[ "$OSBITS" == "x86_64" ]]; then |
|
||||||
OS_NAME="linux64" |
|
||||||
elif [[ "$OSBITS" == "armv7l" || "$OSBITS" == "aarch64" ]]; then |
|
||||||
OS_NAME="linuxarm" |
|
||||||
else |
|
||||||
OS_NAME="$OSTYPE-$OSBITS" |
|
||||||
echo "Unknown OS '$OS_NAME'" |
|
||||||
exit 1 |
|
||||||
fi |
|
||||||
elif [[ "$OSTYPE" == "darwin"* ]]; then |
|
||||||
export OS_IS_MACOS="1" |
|
||||||
ARCHIVE_FORMAT="zip" |
|
||||||
OS_NAME="macosx" |
|
||||||
elif [[ "$OSTYPE" == "cygwin" ]] || [[ "$OSTYPE" == "msys" ]] || [[ "$OSTYPE" == "win32" ]]; then |
|
||||||
export OS_IS_WINDOWS="1" |
|
||||||
ARCHIVE_FORMAT="zip" |
|
||||||
OS_NAME="windows" |
|
||||||
else |
|
||||||
OS_NAME="$OSTYPE-$OSBITS" |
|
||||||
echo "Unknown OS '$OS_NAME'" |
|
||||||
exit 1 |
|
||||||
fi |
|
||||||
export OS_NAME |
|
||||||
|
|
||||||
ARDUINO_BUILD_DIR="$HOME/.arduino/build.tmp" |
|
||||||
ARDUINO_CACHE_DIR="$HOME/.arduino/cache.tmp" |
|
||||||
|
|
||||||
if [ "$OS_IS_MACOS" == "1" ]; then |
|
||||||
export ARDUINO_IDE_PATH="/Applications/Arduino.app/Contents/Java" |
|
||||||
export ARDUINO_USR_PATH="$HOME/Documents/Arduino" |
|
||||||
elif [ "$OS_IS_WINDOWS" == "1" ]; then |
|
||||||
export ARDUINO_IDE_PATH="$HOME/arduino_ide" |
|
||||||
export ARDUINO_USR_PATH="$HOME/Documents/Arduino" |
|
||||||
else |
|
||||||
export ARDUINO_IDE_PATH="$HOME/arduino_ide" |
|
||||||
export ARDUINO_USR_PATH="$HOME/Arduino" |
|
||||||
fi |
|
||||||
|
|
||||||
if [ ! -d "$ARDUINO_IDE_PATH" ]; then |
|
||||||
echo "Installing Arduino IDE on $OS_NAME ..." |
|
||||||
echo "Downloading 'arduino-nightly-$OS_NAME.$ARCHIVE_FORMAT' to 'arduino.$ARCHIVE_FORMAT' ..." |
|
||||||
if [ "$OS_IS_LINUX" == "1" ]; then |
|
||||||
wget -O "arduino.$ARCHIVE_FORMAT" "https://www.arduino.cc/download.php?f=/arduino-nightly-$OS_NAME.$ARCHIVE_FORMAT" > /dev/null 2>&1 |
|
||||||
echo "Extracting 'arduino.$ARCHIVE_FORMAT' ..." |
|
||||||
tar xf "arduino.$ARCHIVE_FORMAT" > /dev/null |
|
||||||
mv arduino-nightly "$ARDUINO_IDE_PATH" |
|
||||||
else |
|
||||||
curl -o "arduino.$ARCHIVE_FORMAT" -L "https://www.arduino.cc/download.php?f=/arduino-nightly-$OS_NAME.$ARCHIVE_FORMAT" > /dev/null 2>&1 |
|
||||||
echo "Extracting 'arduino.$ARCHIVE_FORMAT' ..." |
|
||||||
unzip "arduino.$ARCHIVE_FORMAT" > /dev/null |
|
||||||
if [ "$OS_IS_MACOS" == "1" ]; then |
|
||||||
mv "Arduino.app" "/Applications/Arduino.app" |
|
||||||
else |
|
||||||
mv arduino-nightly "$ARDUINO_IDE_PATH" |
|
||||||
fi |
|
||||||
fi |
|
||||||
rm -rf "arduino.$ARCHIVE_FORMAT" |
|
||||||
|
|
||||||
mkdir -p "$ARDUINO_USR_PATH/libraries" |
|
||||||
mkdir -p "$ARDUINO_USR_PATH/hardware" |
|
||||||
|
|
||||||
echo "Arduino IDE Installed in '$ARDUINO_IDE_PATH'" |
|
||||||
echo "" |
|
||||||
fi |
|
||||||
|
|
||||||
function build_sketch(){ # build_sketch <fqbn> <path-to-ino> <build-flags> [extra-options] |
|
||||||
if [ "$#" -lt 2 ]; then |
|
||||||
echo "ERROR: Illegal number of parameters" |
|
||||||
echo "USAGE: build_sketch <fqbn> <path-to-ino> <build-flags> [extra-options]" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
|
|
||||||
local fqbn="$1" |
|
||||||
local sketch="$2" |
|
||||||
local build_flags="$3" |
|
||||||
local xtra_opts="$4" |
|
||||||
local win_opts="" |
|
||||||
if [ "$OS_IS_WINDOWS" == "1" ]; then |
|
||||||
local ctags_version=`ls "$ARDUINO_IDE_PATH/tools-builder/ctags/"` |
|
||||||
local preprocessor_version=`ls "$ARDUINO_IDE_PATH/tools-builder/arduino-preprocessor/"` |
|
||||||
win_opts="-prefs=runtime.tools.ctags.path=$ARDUINO_IDE_PATH/tools-builder/ctags/$ctags_version -prefs=runtime.tools.arduino-preprocessor.path=$ARDUINO_IDE_PATH/tools-builder/arduino-preprocessor/$preprocessor_version" |
|
||||||
fi |
|
||||||
|
|
||||||
echo "" |
|
||||||
echo "Compiling '"$(basename "$sketch")"' ..." |
|
||||||
mkdir -p "$ARDUINO_BUILD_DIR" |
|
||||||
mkdir -p "$ARDUINO_CACHE_DIR" |
|
||||||
$ARDUINO_IDE_PATH/arduino-builder -compile -logger=human -core-api-version=10810 \ |
|
||||||
-fqbn=$fqbn \ |
|
||||||
-warnings="all" \ |
|
||||||
-tools "$ARDUINO_IDE_PATH/tools-builder" \ |
|
||||||
-tools "$ARDUINO_IDE_PATH/tools" \ |
|
||||||
-built-in-libraries "$ARDUINO_IDE_PATH/libraries" \ |
|
||||||
-hardware "$ARDUINO_IDE_PATH/hardware" \ |
|
||||||
-hardware "$ARDUINO_USR_PATH/hardware" \ |
|
||||||
-libraries "$ARDUINO_USR_PATH/libraries" \ |
|
||||||
-build-cache "$ARDUINO_CACHE_DIR" \ |
|
||||||
-build-path "$ARDUINO_BUILD_DIR" \ |
|
||||||
-prefs=compiler.cpp.extra_flags="$build_flags" \ |
|
||||||
$win_opts $xtra_opts "$sketch" |
|
||||||
} |
|
||||||
|
|
||||||
function count_sketches() # count_sketches <examples-path> |
|
||||||
{ |
|
||||||
local examples="$1" |
|
||||||
rm -rf sketches.txt |
|
||||||
if [ ! -d "$examples" ]; then |
|
||||||
touch sketches.txt |
|
||||||
return 0 |
|
||||||
fi |
|
||||||
local sketches=$(find $examples -name *.ino) |
|
||||||
local sketchnum=0 |
|
||||||
for sketch in $sketches; do |
|
||||||
local sketchdir=$(dirname $sketch) |
|
||||||
local sketchdirname=$(basename $sketchdir) |
|
||||||
local sketchname=$(basename $sketch) |
|
||||||
if [[ "${sketchdirname}.ino" != "$sketchname" ]]; then |
|
||||||
continue |
|
||||||
fi; |
|
||||||
if [[ -f "$sketchdir/.test.skip" ]]; then |
|
||||||
continue |
|
||||||
fi |
|
||||||
echo $sketch >> sketches.txt |
|
||||||
sketchnum=$(($sketchnum + 1)) |
|
||||||
done |
|
||||||
return $sketchnum |
|
||||||
} |
|
||||||
|
|
||||||
function build_sketches() # build_sketches <fqbn> <examples-path> <chunk> <total-chunks> [extra-options] |
|
||||||
{ |
|
||||||
local fqbn=$1 |
|
||||||
local examples=$2 |
|
||||||
local chunk_idex=$3 |
|
||||||
local chunks_num=$4 |
|
||||||
local xtra_opts=$5 |
|
||||||
|
|
||||||
if [ "$#" -lt 2 ]; then |
|
||||||
echo "ERROR: Illegal number of parameters" |
|
||||||
echo "USAGE: build_sketches <fqbn> <examples-path> [<chunk> <total-chunks>] [extra-options]" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
|
|
||||||
if [ "$#" -lt 4 ]; then |
|
||||||
chunk_idex="0" |
|
||||||
chunks_num="1" |
|
||||||
xtra_opts=$3 |
|
||||||
fi |
|
||||||
|
|
||||||
if [ "$chunks_num" -le 0 ]; then |
|
||||||
echo "ERROR: Chunks count must be positive number" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
if [ "$chunk_idex" -ge "$chunks_num" ]; then |
|
||||||
echo "ERROR: Chunk index must be less than chunks count" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
|
|
||||||
set +e |
|
||||||
count_sketches "$examples" |
|
||||||
local sketchcount=$? |
|
||||||
set -e |
|
||||||
local sketches=$(cat sketches.txt) |
|
||||||
rm -rf sketches.txt |
|
||||||
|
|
||||||
local chunk_size=$(( $sketchcount / $chunks_num )) |
|
||||||
local all_chunks=$(( $chunks_num * $chunk_size )) |
|
||||||
if [ "$all_chunks" -lt "$sketchcount" ]; then |
|
||||||
chunk_size=$(( $chunk_size + 1 )) |
|
||||||
fi |
|
||||||
|
|
||||||
local start_index=$(( $chunk_idex * $chunk_size )) |
|
||||||
if [ "$sketchcount" -le "$start_index" ]; then |
|
||||||
echo "Skipping job" |
|
||||||
return 0 |
|
||||||
fi |
|
||||||
|
|
||||||
local end_index=$(( $(( $chunk_idex + 1 )) * $chunk_size )) |
|
||||||
if [ "$end_index" -gt "$sketchcount" ]; then |
|
||||||
end_index=$sketchcount |
|
||||||
fi |
|
||||||
|
|
||||||
local start_num=$(( $start_index + 1 )) |
|
||||||
echo "Found $sketchcount Sketches"; |
|
||||||
echo "Chunk Count : $chunks_num" |
|
||||||
echo "Chunk Size : $chunk_size" |
|
||||||
echo "Start Sketch: $start_num" |
|
||||||
echo "End Sketch : $end_index" |
|
||||||
|
|
||||||
local sketchnum=0 |
|
||||||
for sketch in $sketches; do |
|
||||||
local sketchdir=$(dirname $sketch) |
|
||||||
local sketchdirname=$(basename $sketchdir) |
|
||||||
local sketchname=$(basename $sketch) |
|
||||||
if [ "${sketchdirname}.ino" != "$sketchname" ] \ |
|
||||||
|| [ -f "$sketchdir/.test.skip" ]; then |
|
||||||
continue |
|
||||||
fi |
|
||||||
sketchnum=$(($sketchnum + 1)) |
|
||||||
if [ "$sketchnum" -le "$start_index" ] \ |
|
||||||
|| [ "$sketchnum" -gt "$end_index" ]; then |
|
||||||
continue |
|
||||||
fi |
|
||||||
local sketchBuildFlags="" |
|
||||||
if [ -f "$sketchdir/.test.build_flags" ]; then |
|
||||||
while read line; do |
|
||||||
sketchBuildFlags="$sketchBuildFlags $line" |
|
||||||
done < "$sketchdir/.test.build_flags" |
|
||||||
fi |
|
||||||
build_sketch "$fqbn" "$sketch" "$sketchBuildFlags" "$xtra_opts" |
|
||||||
local result=$? |
|
||||||
if [ $result -ne 0 ]; then |
|
||||||
return $result |
|
||||||
fi |
|
||||||
done |
|
||||||
return 0 |
|
||||||
} |
|
@ -1,140 +0,0 @@ |
|||||||
#!/bin/bash |
|
||||||
|
|
||||||
echo "Installing Python Wheel ..." |
|
||||||
pip install wheel > /dev/null 2>&1 |
|
||||||
|
|
||||||
echo "Installing PlatformIO ..." |
|
||||||
pip install -U platformio > /dev/null 2>&1 |
|
||||||
|
|
||||||
echo "PlatformIO has been installed" |
|
||||||
echo "" |
|
||||||
|
|
||||||
|
|
||||||
function build_pio_sketch(){ # build_pio_sketch <board> <path-to-ino> <build-flags> |
|
||||||
if [ "$#" -lt 3 ]; then |
|
||||||
echo "ERROR: Illegal number of parameters" |
|
||||||
echo "USAGE: build_pio_sketch <board> <path-to-ino> <build-flags>" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
|
|
||||||
local board="$1" |
|
||||||
local sketch="$2" |
|
||||||
local buildFlags="$3" |
|
||||||
local sketch_dir=$(dirname "$sketch") |
|
||||||
echo "" |
|
||||||
echo "Compiling '"$(basename "$sketch")"' ..." |
|
||||||
python -m platformio ci -l '.' --board "$board" "$sketch_dir" --project-option="board_build.partitions = huge_app.csv" --project-option="build_flags=$buildFlags" |
|
||||||
} |
|
||||||
|
|
||||||
function count_sketches() # count_sketches <examples-path> |
|
||||||
{ |
|
||||||
local examples="$1" |
|
||||||
rm -rf sketches.txt |
|
||||||
if [ ! -d "$examples" ]; then |
|
||||||
touch sketches.txt |
|
||||||
return 0 |
|
||||||
fi |
|
||||||
local sketches=$(find $examples -name *.ino) |
|
||||||
local sketchnum=0 |
|
||||||
for sketch in $sketches; do |
|
||||||
local sketchdir=$(dirname $sketch) |
|
||||||
local sketchdirname=$(basename $sketchdir) |
|
||||||
local sketchname=$(basename $sketch) |
|
||||||
if [[ "${sketchdirname}.ino" != "$sketchname" ]]; then |
|
||||||
continue |
|
||||||
fi; |
|
||||||
if [[ -f "$sketchdir/.test.skip" ]]; then |
|
||||||
continue |
|
||||||
fi |
|
||||||
echo $sketch >> sketches.txt |
|
||||||
sketchnum=$(($sketchnum + 1)) |
|
||||||
done |
|
||||||
return $sketchnum |
|
||||||
} |
|
||||||
|
|
||||||
function build_pio_sketches() # build_pio_sketches <board> <examples-path> <chunk> <total-chunks> |
|
||||||
{ |
|
||||||
if [ "$#" -lt 2 ]; then |
|
||||||
echo "ERROR: Illegal number of parameters" |
|
||||||
echo "USAGE: build_pio_sketches <board> <examples-path> [<chunk> <total-chunks>]" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
|
|
||||||
local board=$1 |
|
||||||
local examples=$2 |
|
||||||
local chunk_idex=$3 |
|
||||||
local chunks_num=$4 |
|
||||||
|
|
||||||
if [ "$#" -lt 4 ]; then |
|
||||||
chunk_idex="0" |
|
||||||
chunks_num="1" |
|
||||||
fi |
|
||||||
|
|
||||||
if [ "$chunks_num" -le 0 ]; then |
|
||||||
echo "ERROR: Chunks count must be positive number" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
if [ "$chunk_idex" -ge "$chunks_num" ]; then |
|
||||||
echo "ERROR: Chunk index must be less than chunks count" |
|
||||||
return 1 |
|
||||||
fi |
|
||||||
|
|
||||||
set +e |
|
||||||
count_sketches "$examples" |
|
||||||
local sketchcount=$? |
|
||||||
set -e |
|
||||||
local sketches=$(cat sketches.txt) |
|
||||||
rm -rf sketches.txt |
|
||||||
|
|
||||||
local chunk_size=$(( $sketchcount / $chunks_num )) |
|
||||||
local all_chunks=$(( $chunks_num * $chunk_size )) |
|
||||||
if [ "$all_chunks" -lt "$sketchcount" ]; then |
|
||||||
chunk_size=$(( $chunk_size + 1 )) |
|
||||||
fi |
|
||||||
|
|
||||||
local start_index=$(( $chunk_idex * $chunk_size )) |
|
||||||
if [ "$sketchcount" -le "$start_index" ]; then |
|
||||||
echo "Skipping job" |
|
||||||
return 0 |
|
||||||
fi |
|
||||||
|
|
||||||
local end_index=$(( $(( $chunk_idex + 1 )) * $chunk_size )) |
|
||||||
if [ "$end_index" -gt "$sketchcount" ]; then |
|
||||||
end_index=$sketchcount |
|
||||||
fi |
|
||||||
|
|
||||||
local start_num=$(( $start_index + 1 )) |
|
||||||
echo "Found $sketchcount Sketches"; |
|
||||||
echo "Chunk Count : $chunks_num" |
|
||||||
echo "Chunk Size : $chunk_size" |
|
||||||
echo "Start Sketch: $start_num" |
|
||||||
echo "End Sketch : $end_index" |
|
||||||
|
|
||||||
local sketchnum=0 |
|
||||||
for sketch in $sketches; do |
|
||||||
local sketchdir=$(dirname $sketch) |
|
||||||
local sketchdirname=$(basename $sketchdir) |
|
||||||
local sketchname=$(basename $sketch) |
|
||||||
if [ "${sketchdirname}.ino" != "$sketchname" ] \ |
|
||||||
|| [ -f "$sketchdir/.test.skip" ]; then |
|
||||||
continue |
|
||||||
fi |
|
||||||
local sketchBuildFlags="" |
|
||||||
if [ -f "$sketchdir/.test.build_flags" ]; then |
|
||||||
while read line; do |
|
||||||
sketchBuildFlags="$sketchBuildFlags $line" |
|
||||||
done < "$sketchdir/.test.build_flags" |
|
||||||
fi |
|
||||||
sketchnum=$(($sketchnum + 1)) |
|
||||||
if [ "$sketchnum" -le "$start_index" ] \ |
|
||||||
|| [ "$sketchnum" -gt "$end_index" ]; then |
|
||||||
continue |
|
||||||
fi |
|
||||||
build_pio_sketch "$board" "$sketch" "$sketchBuildFlags" |
|
||||||
local result=$? |
|
||||||
if [ $result -ne 0 ]; then |
|
||||||
return $result |
|
||||||
fi |
|
||||||
done |
|
||||||
return 0 |
|
||||||
} |
|
@ -1,71 +0,0 @@ |
|||||||
#!/bin/bash |
|
||||||
|
|
||||||
set -e |
|
||||||
|
|
||||||
if [ ! -z "$TRAVIS_BUILD_DIR" ]; then |
|
||||||
export GITHUB_WORKSPACE="$TRAVIS_BUILD_DIR" |
|
||||||
export GITHUB_REPOSITORY="$TRAVIS_REPO_SLUG" |
|
||||||
elif [ -z "$GITHUB_WORKSPACE" ]; then |
|
||||||
export GITHUB_WORKSPACE="$PWD" |
|
||||||
export GITHUB_REPOSITORY="me-no-dev/ESPAsyncWebServer" |
|
||||||
fi |
|
||||||
|
|
||||||
TARGET_PLATFORM="$1" |
|
||||||
CHUNK_INDEX=$2 |
|
||||||
CHUNKS_CNT=$3 |
|
||||||
BUILD_PIO=0 |
|
||||||
if [ "$#" -lt 1 ]; then |
|
||||||
TARGET_PLATFORM="esp32" |
|
||||||
fi |
|
||||||
if [ "$#" -lt 3 ] || [ "$CHUNKS_CNT" -le 0 ]; then |
|
||||||
CHUNK_INDEX=0 |
|
||||||
CHUNKS_CNT=1 |
|
||||||
elif [ "$CHUNK_INDEX" -gt "$CHUNKS_CNT" ]; then |
|
||||||
CHUNK_INDEX=$CHUNKS_CNT |
|
||||||
elif [ "$CHUNK_INDEX" -eq "$CHUNKS_CNT" ]; then |
|
||||||
BUILD_PIO=1 |
|
||||||
fi |
|
||||||
|
|
||||||
if [ "$BUILD_PIO" -eq 0 ]; then |
|
||||||
# ArduinoIDE Test |
|
||||||
source ./.github/scripts/install-arduino-ide.sh |
|
||||||
|
|
||||||
echo "Installing ESPAsyncWebServer ..." |
|
||||||
cp -rf "$GITHUB_WORKSPACE" "$ARDUINO_USR_PATH/libraries/ESPAsyncWebServer" |
|
||||||
echo "Installing ArduinoJson ..." |
|
||||||
git clone https://github.com/bblanchon/ArduinoJson "$ARDUINO_USR_PATH/libraries/ArduinoJson" > /dev/null 2>&1 |
|
||||||
|
|
||||||
if [[ "$TARGET_PLATFORM" == "esp32" ]]; then |
|
||||||
echo "Installing AsyncTCP ..." |
|
||||||
git clone https://github.com/me-no-dev/AsyncTCP "$ARDUINO_USR_PATH/libraries/AsyncTCP" > /dev/null 2>&1 |
|
||||||
FQBN="espressif:esp32:esp32:PSRAM=enabled,PartitionScheme=huge_app" |
|
||||||
source ./.github/scripts/install-arduino-core-esp32.sh |
|
||||||
echo "BUILDING ESP32 EXAMPLES" |
|
||||||
else |
|
||||||
echo "Installing ESPAsyncTCP ..." |
|
||||||
git clone https://github.com/me-no-dev/ESPAsyncTCP "$ARDUINO_USR_PATH/libraries/ESPAsyncTCP" > /dev/null 2>&1 |
|
||||||
FQBN="esp8266com:esp8266:generic:eesz=4M1M,ip=lm2f" |
|
||||||
source ./.github/scripts/install-arduino-core-esp8266.sh |
|
||||||
echo "BUILDING ESP8266 EXAMPLES" |
|
||||||
fi |
|
||||||
build_sketches "$FQBN" "$GITHUB_WORKSPACE/examples" "$CHUNK_INDEX" "$CHUNKS_CNT" |
|
||||||
else |
|
||||||
# PlatformIO Test |
|
||||||
source ./.github/scripts/install-platformio.sh |
|
||||||
|
|
||||||
python -m platformio lib --storage-dir "$GITHUB_WORKSPACE" install |
|
||||||
echo "Installing ArduinoJson ..." |
|
||||||
python -m platformio lib -g install https://github.com/bblanchon/ArduinoJson.git > /dev/null 2>&1 |
|
||||||
if [[ "$TARGET_PLATFORM" == "esp32" ]]; then |
|
||||||
BOARD="esp32dev" |
|
||||||
echo "Installing AsyncTCP ..." |
|
||||||
python -m platformio lib -g install https://github.com/me-no-dev/AsyncTCP.git > /dev/null 2>&1 |
|
||||||
echo "BUILDING ESP32 EXAMPLES" |
|
||||||
else |
|
||||||
BOARD="esp12e" |
|
||||||
echo "Installing ESPAsyncTCP ..." |
|
||||||
python -m platformio lib -g install https://github.com/me-no-dev/ESPAsyncTCP.git > /dev/null 2>&1 |
|
||||||
echo "BUILDING ESP8266 EXAMPLES" |
|
||||||
fi |
|
||||||
build_pio_sketches "$BOARD" "$GITHUB_WORKSPACE/examples" |
|
||||||
fi |
|
@ -1,31 +0,0 @@ |
|||||||
# Configuration for probot-stale - https://github.com/probot/stale |
|
||||||
|
|
||||||
daysUntilStale: 60 |
|
||||||
daysUntilClose: 14 |
|
||||||
limitPerRun: 30 |
|
||||||
staleLabel: stale |
|
||||||
exemptLabels: |
|
||||||
- pinned |
|
||||||
- security |
|
||||||
- "to be implemented" |
|
||||||
- "for reference" |
|
||||||
- "move to PR" |
|
||||||
- "enhancement" |
|
||||||
|
|
||||||
only: issues |
|
||||||
onlyLabels: [] |
|
||||||
exemptProjects: false |
|
||||||
exemptMilestones: false |
|
||||||
exemptAssignees: false |
|
||||||
|
|
||||||
markComment: > |
|
||||||
[STALE_SET] This issue has been automatically marked as stale because it has not had |
|
||||||
recent activity. It will be closed in 14 days if no further activity occurs. Thank you |
|
||||||
for your contributions. |
|
||||||
|
|
||||||
unmarkComment: > |
|
||||||
[STALE_CLR] This issue has been removed from the stale queue. Please ensure activity to keep it openin the future. |
|
||||||
|
|
||||||
closeComment: > |
|
||||||
[STALE_DEL] This stale issue has been automatically closed. Thank you for your contributions. |
|
||||||
|
|
@ -1,34 +0,0 @@ |
|||||||
name: ESP Async Web Server CI |
|
||||||
|
|
||||||
on: |
|
||||||
push: |
|
||||||
branches: |
|
||||||
- master |
|
||||||
- release/* |
|
||||||
pull_request: |
|
||||||
|
|
||||||
jobs: |
|
||||||
|
|
||||||
build-arduino: |
|
||||||
name: Arduino for ${{ matrix.board }} on ${{ matrix.os }} |
|
||||||
runs-on: ${{ matrix.os }} |
|
||||||
strategy: |
|
||||||
matrix: |
|
||||||
os: [ubuntu-latest, windows-latest, macOS-latest] |
|
||||||
board: [esp32, esp8266] |
|
||||||
steps: |
|
||||||
- uses: actions/checkout@v1 |
|
||||||
- name: Build Tests |
|
||||||
run: bash ./.github/scripts/on-push.sh ${{ matrix.board }} 0 1 |
|
||||||
|
|
||||||
# build-pio: |
|
||||||
# name: PlatformIO for ${{ matrix.board }} on ${{ matrix.os }} |
|
||||||
# runs-on: ${{ matrix.os }} |
|
||||||
# strategy: |
|
||||||
# matrix: |
|
||||||
# os: [ubuntu-latest, windows-latest, macOS-latest] |
|
||||||
# board: [esp32, esp8266] |
|
||||||
# steps: |
|
||||||
# - uses: actions/checkout@v1 |
|
||||||
# - name: Build Tests |
|
||||||
# run: bash ./.github/scripts/on-push.sh ${{ matrix.board }} 1 1 |
|
@ -1,2 +0,0 @@ |
|||||||
.vscode |
|
||||||
.DS_Store |
|
@ -1,46 +0,0 @@ |
|||||||
sudo: false |
|
||||||
|
|
||||||
language: python |
|
||||||
|
|
||||||
os: |
|
||||||
- linux |
|
||||||
|
|
||||||
git: |
|
||||||
depth: false |
|
||||||
|
|
||||||
stages: |
|
||||||
- build |
|
||||||
|
|
||||||
jobs: |
|
||||||
include: |
|
||||||
|
|
||||||
- name: "Build Arduino ESP32" |
|
||||||
if: tag IS blank AND (type = pull_request OR (type = push AND branch = master)) |
|
||||||
stage: build |
|
||||||
script: bash $TRAVIS_BUILD_DIR/.github/scripts/on-push.sh esp32 |
|
||||||
|
|
||||||
- name: "Build Arduino ESP8266" |
|
||||||
if: tag IS blank AND (type = pull_request OR (type = push AND branch = master)) |
|
||||||
stage: build |
|
||||||
script: bash $TRAVIS_BUILD_DIR/.github/scripts/on-push.sh esp8266 |
|
||||||
|
|
||||||
- name: "Build Platformio ESP32" |
|
||||||
if: tag IS blank AND (type = pull_request OR (type = push AND branch = master)) |
|
||||||
stage: build |
|
||||||
script: bash $TRAVIS_BUILD_DIR/.github/scripts/on-push.sh esp32 1 1 |
|
||||||
|
|
||||||
- name: "Build Platformio ESP8266" |
|
||||||
if: tag IS blank AND (type = pull_request OR (type = push AND branch = master)) |
|
||||||
stage: build |
|
||||||
script: bash $TRAVIS_BUILD_DIR/.github/scripts/on-push.sh esp8266 1 1 |
|
||||||
|
|
||||||
notifications: |
|
||||||
email: |
|
||||||
on_success: change |
|
||||||
on_failure: change |
|
||||||
webhooks: |
|
||||||
urls: |
|
||||||
- https://webhooks.gitter.im/e/60e65d0c78ea0a920347 |
|
||||||
on_success: change # options: [always|never|change] default: always |
|
||||||
on_failure: always # options: [always|never|change] default: always |
|
||||||
on_start: never # options: [always|never|change] default: always |
|
@ -1,17 +0,0 @@ |
|||||||
set(COMPONENT_SRCDIRS |
|
||||||
"src" |
|
||||||
) |
|
||||||
|
|
||||||
set(COMPONENT_ADD_INCLUDEDIRS |
|
||||||
"src" |
|
||||||
) |
|
||||||
|
|
||||||
set(COMPONENT_REQUIRES |
|
||||||
"arduino-esp32" |
|
||||||
"AsyncTCP" |
|
||||||
) |
|
||||||
|
|
||||||
register_component() |
|
||||||
|
|
||||||
target_compile_definitions(${COMPONENT_TARGET} PUBLIC -DESP32) |
|
||||||
target_compile_options(${COMPONENT_TARGET} PRIVATE -fno-rtti) |
|
File diff suppressed because it is too large
Load Diff
@ -1 +0,0 @@ |
|||||||
theme: jekyll-theme-cayman |
|
@ -1,3 +0,0 @@ |
|||||||
COMPONENT_ADD_INCLUDEDIRS := src
|
|
||||||
COMPONENT_SRCDIRS := src
|
|
||||||
CXXFLAGS += -fno-rtti
|
|
@ -1,47 +0,0 @@ |
|||||||
#include <DNSServer.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include <WiFi.h> |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#elif defined(ESP8266) |
|
||||||
#include <ESP8266WiFi.h> |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#endif |
|
||||||
#include "ESPAsyncWebServer.h" |
|
||||||
|
|
||||||
DNSServer dnsServer; |
|
||||||
AsyncWebServer server(80); |
|
||||||
|
|
||||||
class CaptiveRequestHandler : public AsyncWebHandler { |
|
||||||
public: |
|
||||||
CaptiveRequestHandler() {} |
|
||||||
virtual ~CaptiveRequestHandler() {} |
|
||||||
|
|
||||||
bool canHandle(AsyncWebServerRequest *request){ |
|
||||||
//request->addInterestingHeader("ANY");
|
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
void handleRequest(AsyncWebServerRequest *request) { |
|
||||||
AsyncResponseStream *response = request->beginResponseStream("text/html"); |
|
||||||
response->print("<!DOCTYPE html><html><head><title>Captive Portal</title></head><body>"); |
|
||||||
response->print("<p>This is out captive portal front page.</p>"); |
|
||||||
response->printf("<p>You were trying to reach: http://%s%s</p>", request->host().c_str(), request->url().c_str()); |
|
||||||
response->printf("<p>Try opening <a href='http://%s'>this link</a> instead</p>", WiFi.softAPIP().toString().c_str()); |
|
||||||
response->print("</body></html>"); |
|
||||||
request->send(response); |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
|
|
||||||
void setup(){ |
|
||||||
//your other setup stuff...
|
|
||||||
WiFi.softAP("esp-captive"); |
|
||||||
dnsServer.start(53, "*", WiFi.softAPIP()); |
|
||||||
server.addHandler(new CaptiveRequestHandler()).setFilter(ON_AP_FILTER);//only when requested from AP
|
|
||||||
//more handlers...
|
|
||||||
server.begin(); |
|
||||||
} |
|
||||||
|
|
||||||
void loop(){ |
|
||||||
dnsServer.processNextRequest(); |
|
||||||
} |
|
@ -1,221 +0,0 @@ |
|||||||
#include <ArduinoOTA.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include <FS.h> |
|
||||||
#include <SPIFFS.h> |
|
||||||
#include <ESPmDNS.h> |
|
||||||
#include <WiFi.h> |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#elif defined(ESP8266) |
|
||||||
#include <ESP8266WiFi.h> |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#include <ESP8266mDNS.h> |
|
||||||
#endif |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
#include <SPIFFSEditor.h> |
|
||||||
|
|
||||||
// SKETCH BEGIN
|
|
||||||
AsyncWebServer server(80); |
|
||||||
AsyncWebSocket ws("/ws"); |
|
||||||
AsyncEventSource events("/events"); |
|
||||||
|
|
||||||
void onWsEvent(AsyncWebSocket * server, AsyncWebSocketClient * client, AwsEventType type, void * arg, uint8_t *data, size_t len){ |
|
||||||
if(type == WS_EVT_CONNECT){ |
|
||||||
Serial.printf("ws[%s][%u] connect\n", server->url(), client->id()); |
|
||||||
client->printf("Hello Client %u :)", client->id()); |
|
||||||
client->ping(); |
|
||||||
} else if(type == WS_EVT_DISCONNECT){ |
|
||||||
Serial.printf("ws[%s][%u] disconnect\n", server->url(), client->id()); |
|
||||||
} else if(type == WS_EVT_ERROR){ |
|
||||||
Serial.printf("ws[%s][%u] error(%u): %s\n", server->url(), client->id(), *((uint16_t*)arg), (char*)data); |
|
||||||
} else if(type == WS_EVT_PONG){ |
|
||||||
Serial.printf("ws[%s][%u] pong[%u]: %s\n", server->url(), client->id(), len, (len)?(char*)data:""); |
|
||||||
} else if(type == WS_EVT_DATA){ |
|
||||||
AwsFrameInfo * info = (AwsFrameInfo*)arg; |
|
||||||
String msg = ""; |
|
||||||
if(info->final && info->index == 0 && info->len == len){ |
|
||||||
//the whole message is in a single frame and we got all of it's data
|
|
||||||
Serial.printf("ws[%s][%u] %s-message[%llu]: ", server->url(), client->id(), (info->opcode == WS_TEXT)?"text":"binary", info->len); |
|
||||||
|
|
||||||
if(info->opcode == WS_TEXT){ |
|
||||||
for(size_t i=0; i < info->len; i++) { |
|
||||||
msg += (char) data[i]; |
|
||||||
} |
|
||||||
} else { |
|
||||||
char buff[3]; |
|
||||||
for(size_t i=0; i < info->len; i++) { |
|
||||||
sprintf(buff, "%02x ", (uint8_t) data[i]); |
|
||||||
msg += buff ; |
|
||||||
} |
|
||||||
} |
|
||||||
Serial.printf("%s\n",msg.c_str()); |
|
||||||
|
|
||||||
if(info->opcode == WS_TEXT) |
|
||||||
client->text("I got your text message"); |
|
||||||
else |
|
||||||
client->binary("I got your binary message"); |
|
||||||
} else { |
|
||||||
//message is comprised of multiple frames or the frame is split into multiple packets
|
|
||||||
if(info->index == 0){ |
|
||||||
if(info->num == 0) |
|
||||||
Serial.printf("ws[%s][%u] %s-message start\n", server->url(), client->id(), (info->message_opcode == WS_TEXT)?"text":"binary"); |
|
||||||
Serial.printf("ws[%s][%u] frame[%u] start[%llu]\n", server->url(), client->id(), info->num, info->len); |
|
||||||
} |
|
||||||
|
|
||||||
Serial.printf("ws[%s][%u] frame[%u] %s[%llu - %llu]: ", server->url(), client->id(), info->num, (info->message_opcode == WS_TEXT)?"text":"binary", info->index, info->index + len); |
|
||||||
|
|
||||||
if(info->opcode == WS_TEXT){ |
|
||||||
for(size_t i=0; i < len; i++) { |
|
||||||
msg += (char) data[i]; |
|
||||||
} |
|
||||||
} else { |
|
||||||
char buff[3]; |
|
||||||
for(size_t i=0; i < len; i++) { |
|
||||||
sprintf(buff, "%02x ", (uint8_t) data[i]); |
|
||||||
msg += buff ; |
|
||||||
} |
|
||||||
} |
|
||||||
Serial.printf("%s\n",msg.c_str()); |
|
||||||
|
|
||||||
if((info->index + len) == info->len){ |
|
||||||
Serial.printf("ws[%s][%u] frame[%u] end[%llu]\n", server->url(), client->id(), info->num, info->len); |
|
||||||
if(info->final){ |
|
||||||
Serial.printf("ws[%s][%u] %s-message end\n", server->url(), client->id(), (info->message_opcode == WS_TEXT)?"text":"binary"); |
|
||||||
if(info->message_opcode == WS_TEXT) |
|
||||||
client->text("I got your text message"); |
|
||||||
else |
|
||||||
client->binary("I got your binary message"); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
const char* ssid = "*******"; |
|
||||||
const char* password = "*******"; |
|
||||||
const char * hostName = "esp-async"; |
|
||||||
const char* http_username = "admin"; |
|
||||||
const char* http_password = "admin"; |
|
||||||
|
|
||||||
void setup(){ |
|
||||||
Serial.begin(115200); |
|
||||||
Serial.setDebugOutput(true); |
|
||||||
WiFi.mode(WIFI_AP_STA); |
|
||||||
WiFi.softAP(hostName); |
|
||||||
WiFi.begin(ssid, password); |
|
||||||
if (WiFi.waitForConnectResult() != WL_CONNECTED) { |
|
||||||
Serial.printf("STA: Failed!\n"); |
|
||||||
WiFi.disconnect(false); |
|
||||||
delay(1000); |
|
||||||
WiFi.begin(ssid, password); |
|
||||||
} |
|
||||||
|
|
||||||
//Send OTA events to the browser
|
|
||||||
ArduinoOTA.onStart([]() { events.send("Update Start", "ota"); }); |
|
||||||
ArduinoOTA.onEnd([]() { events.send("Update End", "ota"); }); |
|
||||||
ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) { |
|
||||||
char p[32]; |
|
||||||
sprintf(p, "Progress: %u%%\n", (progress/(total/100))); |
|
||||||
events.send(p, "ota"); |
|
||||||
}); |
|
||||||
ArduinoOTA.onError([](ota_error_t error) { |
|
||||||
if(error == OTA_AUTH_ERROR) events.send("Auth Failed", "ota"); |
|
||||||
else if(error == OTA_BEGIN_ERROR) events.send("Begin Failed", "ota"); |
|
||||||
else if(error == OTA_CONNECT_ERROR) events.send("Connect Failed", "ota"); |
|
||||||
else if(error == OTA_RECEIVE_ERROR) events.send("Recieve Failed", "ota"); |
|
||||||
else if(error == OTA_END_ERROR) events.send("End Failed", "ota"); |
|
||||||
}); |
|
||||||
ArduinoOTA.setHostname(hostName); |
|
||||||
ArduinoOTA.begin(); |
|
||||||
|
|
||||||
MDNS.addService("http","tcp",80); |
|
||||||
|
|
||||||
SPIFFS.begin(); |
|
||||||
|
|
||||||
ws.onEvent(onWsEvent); |
|
||||||
server.addHandler(&ws); |
|
||||||
|
|
||||||
events.onConnect([](AsyncEventSourceClient *client){ |
|
||||||
client->send("hello!",NULL,millis(),1000); |
|
||||||
}); |
|
||||||
server.addHandler(&events); |
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
server.addHandler(new SPIFFSEditor(SPIFFS, http_username,http_password)); |
|
||||||
#elif defined(ESP8266) |
|
||||||
server.addHandler(new SPIFFSEditor(http_username,http_password)); |
|
||||||
#endif |
|
||||||
|
|
||||||
server.on("/heap", HTTP_GET, [](AsyncWebServerRequest *request){ |
|
||||||
request->send(200, "text/plain", String(ESP.getFreeHeap())); |
|
||||||
}); |
|
||||||
|
|
||||||
server.serveStatic("/", SPIFFS, "/").setDefaultFile("index.htm"); |
|
||||||
|
|
||||||
server.onNotFound([](AsyncWebServerRequest *request){ |
|
||||||
Serial.printf("NOT_FOUND: "); |
|
||||||
if(request->method() == HTTP_GET) |
|
||||||
Serial.printf("GET"); |
|
||||||
else if(request->method() == HTTP_POST) |
|
||||||
Serial.printf("POST"); |
|
||||||
else if(request->method() == HTTP_DELETE) |
|
||||||
Serial.printf("DELETE"); |
|
||||||
else if(request->method() == HTTP_PUT) |
|
||||||
Serial.printf("PUT"); |
|
||||||
else if(request->method() == HTTP_PATCH) |
|
||||||
Serial.printf("PATCH"); |
|
||||||
else if(request->method() == HTTP_HEAD) |
|
||||||
Serial.printf("HEAD"); |
|
||||||
else if(request->method() == HTTP_OPTIONS) |
|
||||||
Serial.printf("OPTIONS"); |
|
||||||
else |
|
||||||
Serial.printf("UNKNOWN"); |
|
||||||
Serial.printf(" http://%s%s\n", request->host().c_str(), request->url().c_str()); |
|
||||||
|
|
||||||
if(request->contentLength()){ |
|
||||||
Serial.printf("_CONTENT_TYPE: %s\n", request->contentType().c_str()); |
|
||||||
Serial.printf("_CONTENT_LENGTH: %u\n", request->contentLength()); |
|
||||||
} |
|
||||||
|
|
||||||
int headers = request->headers(); |
|
||||||
int i; |
|
||||||
for(i=0;i<headers;i++){ |
|
||||||
AsyncWebHeader* h = request->getHeader(i); |
|
||||||
Serial.printf("_HEADER[%s]: %s\n", h->name().c_str(), h->value().c_str()); |
|
||||||
} |
|
||||||
|
|
||||||
int params = request->params(); |
|
||||||
for(i=0;i<params;i++){ |
|
||||||
AsyncWebParameter* p = request->getParam(i); |
|
||||||
if(p->isFile()){ |
|
||||||
Serial.printf("_FILE[%s]: %s, size: %u\n", p->name().c_str(), p->value().c_str(), p->size()); |
|
||||||
} else if(p->isPost()){ |
|
||||||
Serial.printf("_POST[%s]: %s\n", p->name().c_str(), p->value().c_str()); |
|
||||||
} else { |
|
||||||
Serial.printf("_GET[%s]: %s\n", p->name().c_str(), p->value().c_str()); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
request->send(404); |
|
||||||
}); |
|
||||||
server.onFileUpload([](AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final){ |
|
||||||
if(!index) |
|
||||||
Serial.printf("UploadStart: %s\n", filename.c_str()); |
|
||||||
Serial.printf("%s", (const char*)data); |
|
||||||
if(final) |
|
||||||
Serial.printf("UploadEnd: %s (%u)\n", filename.c_str(), index+len); |
|
||||||
}); |
|
||||||
server.onRequestBody([](AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total){ |
|
||||||
if(!index) |
|
||||||
Serial.printf("BodyStart: %u\n", total); |
|
||||||
Serial.printf("%s", (const char*)data); |
|
||||||
if(index + len == total) |
|
||||||
Serial.printf("BodyEnd: %u\n", total); |
|
||||||
}); |
|
||||||
server.begin(); |
|
||||||
} |
|
||||||
|
|
||||||
void loop(){ |
|
||||||
ArduinoOTA.handle(); |
|
||||||
ws.cleanupClients(); |
|
||||||
} |
|
@ -1,2 +0,0 @@ |
|||||||
/*.js.gz |
|
||||||
/.exclude.files |
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 1.1 KiB |
@ -1,131 +0,0 @@ |
|||||||
<!-- |
|
||||||
FSWebServer - Example Index Page |
|
||||||
Copyright (c) 2015 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the ESP8266WebServer library for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
--> |
|
||||||
<!DOCTYPE html> |
|
||||||
<html> |
|
||||||
<head> |
|
||||||
<meta http-equiv="Content-type" content="text/html; charset=utf-8"> |
|
||||||
<title>WebSocketTester</title> |
|
||||||
<style type="text/css" media="screen"> |
|
||||||
body { |
|
||||||
margin:0; |
|
||||||
padding:0; |
|
||||||
background-color: black; |
|
||||||
} |
|
||||||
|
|
||||||
#dbg, #input_div, #input_el { |
|
||||||
font-family: monaco; |
|
||||||
font-size: 12px; |
|
||||||
line-height: 13px; |
|
||||||
color: #AAA; |
|
||||||
} |
|
||||||
|
|
||||||
#dbg, #input_div { |
|
||||||
margin:0; |
|
||||||
padding:0; |
|
||||||
padding-left:4px; |
|
||||||
} |
|
||||||
|
|
||||||
#input_el { |
|
||||||
width:98%; |
|
||||||
background-color: rgba(0,0,0,0); |
|
||||||
border: 0px; |
|
||||||
} |
|
||||||
#input_el:focus { |
|
||||||
outline: none; |
|
||||||
} |
|
||||||
</style> |
|
||||||
<script type="text/javascript"> |
|
||||||
var ws = null; |
|
||||||
function ge(s){ return document.getElementById(s);} |
|
||||||
function ce(s){ return document.createElement(s);} |
|
||||||
function stb(){ window.scrollTo(0, document.body.scrollHeight || document.documentElement.scrollHeight); } |
|
||||||
function sendBlob(str){ |
|
||||||
var buf = new Uint8Array(str.length); |
|
||||||
for (var i = 0; i < str.length; ++i) buf[i] = str.charCodeAt(i); |
|
||||||
ws.send(buf); |
|
||||||
} |
|
||||||
function addMessage(m){ |
|
||||||
var msg = ce("div"); |
|
||||||
msg.innerText = m; |
|
||||||
ge("dbg").appendChild(msg); |
|
||||||
stb(); |
|
||||||
} |
|
||||||
function startSocket(){ |
|
||||||
ws = new WebSocket('ws://'+document.location.host+'/ws',['arduino']); |
|
||||||
ws.binaryType = "arraybuffer"; |
|
||||||
ws.onopen = function(e){ |
|
||||||
addMessage("Connected"); |
|
||||||
}; |
|
||||||
ws.onclose = function(e){ |
|
||||||
addMessage("Disconnected"); |
|
||||||
}; |
|
||||||
ws.onerror = function(e){ |
|
||||||
console.log("ws error", e); |
|
||||||
addMessage("Error"); |
|
||||||
}; |
|
||||||
ws.onmessage = function(e){ |
|
||||||
var msg = ""; |
|
||||||
if(e.data instanceof ArrayBuffer){ |
|
||||||
msg = "BIN:"; |
|
||||||
var bytes = new Uint8Array(e.data); |
|
||||||
for (var i = 0; i < bytes.length; i++) { |
|
||||||
msg += String.fromCharCode(bytes[i]); |
|
||||||
} |
|
||||||
} else { |
|
||||||
msg = "TXT:"+e.data; |
|
||||||
} |
|
||||||
addMessage(msg); |
|
||||||
}; |
|
||||||
ge("input_el").onkeydown = function(e){ |
|
||||||
stb(); |
|
||||||
if(e.keyCode == 13 && ge("input_el").value != ""){ |
|
||||||
ws.send(ge("input_el").value); |
|
||||||
ge("input_el").value = ""; |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
function startEvents(){ |
|
||||||
var es = new EventSource('/events'); |
|
||||||
es.onopen = function(e) { |
|
||||||
addMessage("Events Opened"); |
|
||||||
}; |
|
||||||
es.onerror = function(e) { |
|
||||||
if (e.target.readyState != EventSource.OPEN) { |
|
||||||
addMessage("Events Closed"); |
|
||||||
} |
|
||||||
}; |
|
||||||
es.onmessage = function(e) { |
|
||||||
addMessage("Event: " + e.data); |
|
||||||
}; |
|
||||||
es.addEventListener('ota', function(e) { |
|
||||||
addMessage("Event[ota]: " + e.data); |
|
||||||
}, false); |
|
||||||
} |
|
||||||
function onBodyLoad(){ |
|
||||||
startSocket(); |
|
||||||
startEvents(); |
|
||||||
} |
|
||||||
</script> |
|
||||||
</head> |
|
||||||
<body id="body" onload="onBodyLoad()"> |
|
||||||
<pre id="dbg"></pre> |
|
||||||
<div id="input_div"> |
|
||||||
$<input type="text" value="" id="input_el"> |
|
||||||
</div> |
|
||||||
</body> |
|
||||||
</html> |
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -1 +0,0 @@ |
|||||||
-DASYNCWEBSERVER_REGEX=1 |
|
@ -1,77 +0,0 @@ |
|||||||
//
|
|
||||||
// A simple server implementation with regex routes:
|
|
||||||
// * serve static messages
|
|
||||||
// * read GET and POST parameters
|
|
||||||
// * handle missing pages / 404s
|
|
||||||
//
|
|
||||||
|
|
||||||
// Add buildflag ASYNCWEBSERVER_REGEX to enable the regex support
|
|
||||||
|
|
||||||
// For platformio: platformio.ini:
|
|
||||||
// build_flags =
|
|
||||||
// -DASYNCWEBSERVER_REGEX
|
|
||||||
|
|
||||||
// For arduino IDE: create/update platform.local.txt
|
|
||||||
// Windows: C:\Users\(username)\AppData\Local\Arduino15\packages\espxxxx\hardware\espxxxx\{version}\platform.local.txt
|
|
||||||
// Linux: ~/.arduino15/packages/espxxxx/hardware/espxxxx/{version}/platform.local.txt
|
|
||||||
//
|
|
||||||
// compiler.cpp.extra_flags=-DASYNCWEBSERVER_REGEX=1
|
|
||||||
|
|
||||||
#include <Arduino.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include <WiFi.h> |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#elif defined(ESP8266) |
|
||||||
#include <ESP8266WiFi.h> |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#endif |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
|
|
||||||
AsyncWebServer server(80); |
|
||||||
|
|
||||||
const char* ssid = "YOUR_SSID"; |
|
||||||
const char* password = "YOUR_PASSWORD"; |
|
||||||
|
|
||||||
const char* PARAM_MESSAGE = "message"; |
|
||||||
|
|
||||||
void notFound(AsyncWebServerRequest *request) { |
|
||||||
request->send(404, "text/plain", "Not found"); |
|
||||||
} |
|
||||||
|
|
||||||
void setup() { |
|
||||||
|
|
||||||
Serial.begin(115200); |
|
||||||
WiFi.mode(WIFI_STA); |
|
||||||
WiFi.begin(ssid, password); |
|
||||||
if (WiFi.waitForConnectResult() != WL_CONNECTED) { |
|
||||||
Serial.printf("WiFi Failed!\n"); |
|
||||||
return; |
|
||||||
} |
|
||||||
|
|
||||||
Serial.print("IP Address: "); |
|
||||||
Serial.println(WiFi.localIP()); |
|
||||||
|
|
||||||
server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ |
|
||||||
request->send(200, "text/plain", "Hello, world"); |
|
||||||
}); |
|
||||||
|
|
||||||
// Send a GET request to <IP>/sensor/<number>
|
|
||||||
server.on("^\\/sensor\\/([0-9]+)$", HTTP_GET, [] (AsyncWebServerRequest *request) { |
|
||||||
String sensorNumber = request->pathArg(0); |
|
||||||
request->send(200, "text/plain", "Hello, sensor: " + sensorNumber); |
|
||||||
}); |
|
||||||
|
|
||||||
// Send a GET request to <IP>/sensor/<number>/action/<action>
|
|
||||||
server.on("^\\/sensor\\/([0-9]+)\\/action\\/([a-zA-Z0-9]+)$", HTTP_GET, [] (AsyncWebServerRequest *request) { |
|
||||||
String sensorNumber = request->pathArg(0); |
|
||||||
String action = request->pathArg(1); |
|
||||||
request->send(200, "text/plain", "Hello, sensor: " + sensorNumber + ", with action: " + action); |
|
||||||
}); |
|
||||||
|
|
||||||
server.onNotFound(notFound); |
|
||||||
|
|
||||||
server.begin(); |
|
||||||
} |
|
||||||
|
|
||||||
void loop() { |
|
||||||
} |
|
@ -1,74 +0,0 @@ |
|||||||
//
|
|
||||||
// A simple server implementation showing how to:
|
|
||||||
// * serve static messages
|
|
||||||
// * read GET and POST parameters
|
|
||||||
// * handle missing pages / 404s
|
|
||||||
//
|
|
||||||
|
|
||||||
#include <Arduino.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include <WiFi.h> |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#elif defined(ESP8266) |
|
||||||
#include <ESP8266WiFi.h> |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#endif |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
|
|
||||||
AsyncWebServer server(80); |
|
||||||
|
|
||||||
const char* ssid = "YOUR_SSID"; |
|
||||||
const char* password = "YOUR_PASSWORD"; |
|
||||||
|
|
||||||
const char* PARAM_MESSAGE = "message"; |
|
||||||
|
|
||||||
void notFound(AsyncWebServerRequest *request) { |
|
||||||
request->send(404, "text/plain", "Not found"); |
|
||||||
} |
|
||||||
|
|
||||||
void setup() { |
|
||||||
|
|
||||||
Serial.begin(115200); |
|
||||||
WiFi.mode(WIFI_STA); |
|
||||||
WiFi.begin(ssid, password); |
|
||||||
if (WiFi.waitForConnectResult() != WL_CONNECTED) { |
|
||||||
Serial.printf("WiFi Failed!\n"); |
|
||||||
return; |
|
||||||
} |
|
||||||
|
|
||||||
Serial.print("IP Address: "); |
|
||||||
Serial.println(WiFi.localIP()); |
|
||||||
|
|
||||||
server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ |
|
||||||
request->send(200, "text/plain", "Hello, world"); |
|
||||||
}); |
|
||||||
|
|
||||||
// Send a GET request to <IP>/get?message=<message>
|
|
||||||
server.on("/get", HTTP_GET, [] (AsyncWebServerRequest *request) { |
|
||||||
String message; |
|
||||||
if (request->hasParam(PARAM_MESSAGE)) { |
|
||||||
message = request->getParam(PARAM_MESSAGE)->value(); |
|
||||||
} else { |
|
||||||
message = "No message sent"; |
|
||||||
} |
|
||||||
request->send(200, "text/plain", "Hello, GET: " + message); |
|
||||||
}); |
|
||||||
|
|
||||||
// Send a POST request to <IP>/post with a form field message set to <message>
|
|
||||||
server.on("/post", HTTP_POST, [](AsyncWebServerRequest *request){ |
|
||||||
String message; |
|
||||||
if (request->hasParam(PARAM_MESSAGE, true)) { |
|
||||||
message = request->getParam(PARAM_MESSAGE, true)->value(); |
|
||||||
} else { |
|
||||||
message = "No message sent"; |
|
||||||
} |
|
||||||
request->send(200, "text/plain", "Hello, POST: " + message); |
|
||||||
}); |
|
||||||
|
|
||||||
server.onNotFound(notFound); |
|
||||||
|
|
||||||
server.begin(); |
|
||||||
} |
|
||||||
|
|
||||||
void loop() { |
|
||||||
} |
|
@ -1,3 +0,0 @@ |
|||||||
JsonArray KEYWORD1 |
|
||||||
add KEYWORD2 |
|
||||||
createArray KEYWORD3 |
|
@ -1,33 +0,0 @@ |
|||||||
{ |
|
||||||
"name":"ESP Async WebServer", |
|
||||||
"description":"Asynchronous HTTP and WebSocket Server Library for ESP8266 and ESP32", |
|
||||||
"keywords":"http,async,websocket,webserver", |
|
||||||
"authors": |
|
||||||
{ |
|
||||||
"name": "Hristo Gochkov", |
|
||||||
"maintainer": true |
|
||||||
}, |
|
||||||
"repository": |
|
||||||
{ |
|
||||||
"type": "git", |
|
||||||
"url": "https://github.com/me-no-dev/ESPAsyncWebServer.git" |
|
||||||
}, |
|
||||||
"version": "1.2.3", |
|
||||||
"license": "LGPL-3.0", |
|
||||||
"frameworks": "arduino", |
|
||||||
"platforms": ["espressif8266", "espressif32"], |
|
||||||
"dependencies": [ |
|
||||||
{ |
|
||||||
"name": "ESPAsyncTCP", |
|
||||||
"platforms": "espressif8266" |
|
||||||
}, |
|
||||||
{ |
|
||||||
"name": "AsyncTCP", |
|
||||||
"platforms": "espressif32" |
|
||||||
}, |
|
||||||
{ |
|
||||||
"name": "Hash", |
|
||||||
"platforms": "espressif8266" |
|
||||||
} |
|
||||||
] |
|
||||||
} |
|
@ -1,9 +0,0 @@ |
|||||||
name=ESP Async WebServer |
|
||||||
version=1.2.3 |
|
||||||
author=Me-No-Dev |
|
||||||
maintainer=Me-No-Dev |
|
||||||
sentence=Async Web Server for ESP8266 and ESP31B |
|
||||||
paragraph=Async Web Server for ESP8266 and ESP31B |
|
||||||
category=Other |
|
||||||
url=https://github.com/me-no-dev/ESPAsyncWebServer |
|
||||||
architectures=* |
|
@ -1,368 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#include "Arduino.h" |
|
||||||
#include "AsyncEventSource.h" |
|
||||||
|
|
||||||
static String generateEventMessage(const char *message, const char *event, uint32_t id, uint32_t reconnect){ |
|
||||||
String ev = ""; |
|
||||||
|
|
||||||
if(reconnect){ |
|
||||||
ev += "retry: "; |
|
||||||
ev += String(reconnect); |
|
||||||
ev += "\r\n"; |
|
||||||
} |
|
||||||
|
|
||||||
if(id){ |
|
||||||
ev += "id: "; |
|
||||||
ev += String(id); |
|
||||||
ev += "\r\n"; |
|
||||||
} |
|
||||||
|
|
||||||
if(event != NULL){ |
|
||||||
ev += "event: "; |
|
||||||
ev += String(event); |
|
||||||
ev += "\r\n"; |
|
||||||
} |
|
||||||
|
|
||||||
if(message != NULL){ |
|
||||||
size_t messageLen = strlen(message); |
|
||||||
char * lineStart = (char *)message; |
|
||||||
char * lineEnd; |
|
||||||
do { |
|
||||||
char * nextN = strchr(lineStart, '\n'); |
|
||||||
char * nextR = strchr(lineStart, '\r'); |
|
||||||
if(nextN == NULL && nextR == NULL){ |
|
||||||
size_t llen = ((char *)message + messageLen) - lineStart; |
|
||||||
char * ldata = (char *)malloc(llen+1); |
|
||||||
if(ldata != NULL){ |
|
||||||
memcpy(ldata, lineStart, llen); |
|
||||||
ldata[llen] = 0; |
|
||||||
ev += "data: "; |
|
||||||
ev += ldata; |
|
||||||
ev += "\r\n\r\n"; |
|
||||||
free(ldata); |
|
||||||
} |
|
||||||
lineStart = (char *)message + messageLen; |
|
||||||
} else { |
|
||||||
char * nextLine = NULL; |
|
||||||
if(nextN != NULL && nextR != NULL){ |
|
||||||
if(nextR < nextN){ |
|
||||||
lineEnd = nextR; |
|
||||||
if(nextN == (nextR + 1)) |
|
||||||
nextLine = nextN + 1; |
|
||||||
else |
|
||||||
nextLine = nextR + 1; |
|
||||||
} else { |
|
||||||
lineEnd = nextN; |
|
||||||
if(nextR == (nextN + 1)) |
|
||||||
nextLine = nextR + 1; |
|
||||||
else |
|
||||||
nextLine = nextN + 1; |
|
||||||
} |
|
||||||
} else if(nextN != NULL){ |
|
||||||
lineEnd = nextN; |
|
||||||
nextLine = nextN + 1; |
|
||||||
} else { |
|
||||||
lineEnd = nextR; |
|
||||||
nextLine = nextR + 1; |
|
||||||
} |
|
||||||
|
|
||||||
size_t llen = lineEnd - lineStart; |
|
||||||
char * ldata = (char *)malloc(llen+1); |
|
||||||
if(ldata != NULL){ |
|
||||||
memcpy(ldata, lineStart, llen); |
|
||||||
ldata[llen] = 0; |
|
||||||
ev += "data: "; |
|
||||||
ev += ldata; |
|
||||||
ev += "\r\n"; |
|
||||||
free(ldata); |
|
||||||
} |
|
||||||
lineStart = nextLine; |
|
||||||
if(lineStart == ((char *)message + messageLen)) |
|
||||||
ev += "\r\n"; |
|
||||||
} |
|
||||||
} while(lineStart < ((char *)message + messageLen)); |
|
||||||
} |
|
||||||
|
|
||||||
return ev; |
|
||||||
} |
|
||||||
|
|
||||||
// Message
|
|
||||||
|
|
||||||
AsyncEventSourceMessage::AsyncEventSourceMessage(const char * data, size_t len) |
|
||||||
: _data(nullptr), _len(len), _sent(0), _acked(0) |
|
||||||
{ |
|
||||||
_data = (uint8_t*)malloc(_len+1); |
|
||||||
if(_data == nullptr){ |
|
||||||
_len = 0; |
|
||||||
} else { |
|
||||||
memcpy(_data, data, len); |
|
||||||
_data[_len] = 0; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
AsyncEventSourceMessage::~AsyncEventSourceMessage() { |
|
||||||
if(_data != NULL) |
|
||||||
free(_data); |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncEventSourceMessage::ack(size_t len, uint32_t time) { |
|
||||||
(void)time; |
|
||||||
// If the whole message is now acked...
|
|
||||||
if(_acked + len > _len){ |
|
||||||
// Return the number of extra bytes acked (they will be carried on to the next message)
|
|
||||||
const size_t extra = _acked + len - _len; |
|
||||||
_acked = _len; |
|
||||||
return extra; |
|
||||||
} |
|
||||||
// Return that no extra bytes left.
|
|
||||||
_acked += len; |
|
||||||
return 0; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncEventSourceMessage::send(AsyncClient *client) { |
|
||||||
const size_t len = _len - _sent; |
|
||||||
if(client->space() < len){ |
|
||||||
return 0; |
|
||||||
} |
|
||||||
size_t sent = client->add((const char *)_data, len); |
|
||||||
if(client->canSend()) |
|
||||||
client->send(); |
|
||||||
_sent += sent; |
|
||||||
return sent;
|
|
||||||
} |
|
||||||
|
|
||||||
// Client
|
|
||||||
|
|
||||||
AsyncEventSourceClient::AsyncEventSourceClient(AsyncWebServerRequest *request, AsyncEventSource *server) |
|
||||||
: _messageQueue(LinkedList<AsyncEventSourceMessage *>([](AsyncEventSourceMessage *m){ delete m; })) |
|
||||||
{ |
|
||||||
_client = request->client(); |
|
||||||
_server = server; |
|
||||||
_lastId = 0; |
|
||||||
if(request->hasHeader("Last-Event-ID")) |
|
||||||
_lastId = atoi(request->getHeader("Last-Event-ID")->value().c_str()); |
|
||||||
|
|
||||||
_client->setRxTimeout(0); |
|
||||||
_client->onError(NULL, NULL); |
|
||||||
_client->onAck([](void *r, AsyncClient* c, size_t len, uint32_t time){ (void)c; ((AsyncEventSourceClient*)(r))->_onAck(len, time); }, this); |
|
||||||
_client->onPoll([](void *r, AsyncClient* c){ (void)c; ((AsyncEventSourceClient*)(r))->_onPoll(); }, this); |
|
||||||
_client->onData(NULL, NULL); |
|
||||||
_client->onTimeout([this](void *r, AsyncClient* c __attribute__((unused)), uint32_t time){ ((AsyncEventSourceClient*)(r))->_onTimeout(time); }, this); |
|
||||||
_client->onDisconnect([this](void *r, AsyncClient* c){ ((AsyncEventSourceClient*)(r))->_onDisconnect(); delete c; }, this); |
|
||||||
|
|
||||||
_server->_addClient(this); |
|
||||||
delete request; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncEventSourceClient::~AsyncEventSourceClient(){ |
|
||||||
_messageQueue.free(); |
|
||||||
close(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::_queueMessage(AsyncEventSourceMessage *dataMessage){ |
|
||||||
if(dataMessage == NULL) |
|
||||||
return; |
|
||||||
if(!connected()){ |
|
||||||
delete dataMessage; |
|
||||||
return; |
|
||||||
} |
|
||||||
if(_messageQueue.length() >= SSE_MAX_QUEUED_MESSAGES){ |
|
||||||
ets_printf("ERROR: Too many messages queued\n"); |
|
||||||
delete dataMessage; |
|
||||||
} else { |
|
||||||
_messageQueue.add(dataMessage); |
|
||||||
} |
|
||||||
if(_client->canSend()) |
|
||||||
_runQueue(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::_onAck(size_t len, uint32_t time){ |
|
||||||
while(len && !_messageQueue.isEmpty()){ |
|
||||||
len = _messageQueue.front()->ack(len, time); |
|
||||||
if(_messageQueue.front()->finished()) |
|
||||||
_messageQueue.remove(_messageQueue.front()); |
|
||||||
} |
|
||||||
|
|
||||||
_runQueue(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::_onPoll(){ |
|
||||||
if(!_messageQueue.isEmpty()){ |
|
||||||
_runQueue(); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
void AsyncEventSourceClient::_onTimeout(uint32_t time __attribute__((unused))){ |
|
||||||
_client->close(true); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::_onDisconnect(){ |
|
||||||
_client = NULL; |
|
||||||
_server->_handleDisconnect(this); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::close(){ |
|
||||||
if(_client != NULL) |
|
||||||
_client->close(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::write(const char * message, size_t len){ |
|
||||||
_queueMessage(new AsyncEventSourceMessage(message, len)); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::send(const char *message, const char *event, uint32_t id, uint32_t reconnect){ |
|
||||||
String ev = generateEventMessage(message, event, id, reconnect); |
|
||||||
_queueMessage(new AsyncEventSourceMessage(ev.c_str(), ev.length())); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceClient::_runQueue(){ |
|
||||||
while(!_messageQueue.isEmpty() && _messageQueue.front()->finished()){ |
|
||||||
_messageQueue.remove(_messageQueue.front()); |
|
||||||
} |
|
||||||
|
|
||||||
for(auto i = _messageQueue.begin(); i != _messageQueue.end(); ++i) |
|
||||||
{ |
|
||||||
if(!(*i)->sent()) |
|
||||||
(*i)->send(_client); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
// Handler
|
|
||||||
|
|
||||||
AsyncEventSource::AsyncEventSource(const String& url) |
|
||||||
: _url(url) |
|
||||||
, _clients(LinkedList<AsyncEventSourceClient *>([](AsyncEventSourceClient *c){ delete c; })) |
|
||||||
, _connectcb(NULL) |
|
||||||
{} |
|
||||||
|
|
||||||
AsyncEventSource::~AsyncEventSource(){ |
|
||||||
close(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSource::onConnect(ArEventHandlerFunction cb){ |
|
||||||
_connectcb = cb; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSource::_addClient(AsyncEventSourceClient * client){ |
|
||||||
/*char * temp = (char *)malloc(2054);
|
|
||||||
if(temp != NULL){ |
|
||||||
memset(temp+1,' ',2048); |
|
||||||
temp[0] = ':'; |
|
||||||
temp[2049] = '\r'; |
|
||||||
temp[2050] = '\n'; |
|
||||||
temp[2051] = '\r'; |
|
||||||
temp[2052] = '\n'; |
|
||||||
temp[2053] = 0; |
|
||||||
client->write((const char *)temp, 2053); |
|
||||||
free(temp); |
|
||||||
}*/ |
|
||||||
|
|
||||||
_clients.add(client); |
|
||||||
if(_connectcb) |
|
||||||
_connectcb(client); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSource::_handleDisconnect(AsyncEventSourceClient * client){ |
|
||||||
_clients.remove(client); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSource::close(){ |
|
||||||
for(const auto &c: _clients){ |
|
||||||
if(c->connected()) |
|
||||||
c->close(); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// pmb fix
|
|
||||||
size_t AsyncEventSource::avgPacketsWaiting() const { |
|
||||||
if(_clients.isEmpty()) |
|
||||||
return 0; |
|
||||||
|
|
||||||
size_t aql=0; |
|
||||||
uint32_t nConnectedClients=0; |
|
||||||
|
|
||||||
for(const auto &c: _clients){ |
|
||||||
if(c->connected()) { |
|
||||||
aql+=c->packetsWaiting(); |
|
||||||
++nConnectedClients; |
|
||||||
} |
|
||||||
} |
|
||||||
// return aql / nConnectedClients;
|
|
||||||
return ((aql) + (nConnectedClients/2))/(nConnectedClients); // round up
|
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSource::send(const char *message, const char *event, uint32_t id, uint32_t reconnect){ |
|
||||||
|
|
||||||
|
|
||||||
String ev = generateEventMessage(message, event, id, reconnect); |
|
||||||
for(const auto &c: _clients){ |
|
||||||
if(c->connected()) { |
|
||||||
c->write(ev.c_str(), ev.length()); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncEventSource::count() const { |
|
||||||
return _clients.count_if([](AsyncEventSourceClient *c){ |
|
||||||
return c->connected(); |
|
||||||
}); |
|
||||||
} |
|
||||||
|
|
||||||
bool AsyncEventSource::canHandle(AsyncWebServerRequest *request){ |
|
||||||
if(request->method() != HTTP_GET || !request->url().equals(_url)) { |
|
||||||
return false; |
|
||||||
} |
|
||||||
request->addInterestingHeader("Last-Event-ID"); |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSource::handleRequest(AsyncWebServerRequest *request){ |
|
||||||
if((_username != "" && _password != "") && !request->authenticate(_username.c_str(), _password.c_str())) |
|
||||||
return request->requestAuthentication(); |
|
||||||
request->send(new AsyncEventSourceResponse(this)); |
|
||||||
} |
|
||||||
|
|
||||||
// Response
|
|
||||||
|
|
||||||
AsyncEventSourceResponse::AsyncEventSourceResponse(AsyncEventSource *server){ |
|
||||||
_server = server; |
|
||||||
_code = 200; |
|
||||||
_contentType = "text/event-stream"; |
|
||||||
_sendContentLength = false; |
|
||||||
addHeader("Cache-Control", "no-cache"); |
|
||||||
addHeader("Connection","keep-alive"); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncEventSourceResponse::_respond(AsyncWebServerRequest *request){ |
|
||||||
String out = _assembleHead(request->version()); |
|
||||||
request->client()->write(out.c_str(), _headLength); |
|
||||||
_state = RESPONSE_WAIT_ACK; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncEventSourceResponse::_ack(AsyncWebServerRequest *request, size_t len, uint32_t time __attribute__((unused))){ |
|
||||||
if(len){ |
|
||||||
new AsyncEventSourceClient(request, _server); |
|
||||||
} |
|
||||||
return 0; |
|
||||||
} |
|
||||||
|
|
@ -1,133 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#ifndef ASYNCEVENTSOURCE_H_ |
|
||||||
#define ASYNCEVENTSOURCE_H_ |
|
||||||
|
|
||||||
#include <Arduino.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#define SSE_MAX_QUEUED_MESSAGES 32 |
|
||||||
#else |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#define SSE_MAX_QUEUED_MESSAGES 8 |
|
||||||
#endif |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
|
|
||||||
#include "AsyncWebSynchronization.h" |
|
||||||
|
|
||||||
#ifdef ESP8266 |
|
||||||
#include <Hash.h> |
|
||||||
#ifdef CRYPTO_HASH_h // include Hash.h from espressif framework if the first include was from the crypto library
|
|
||||||
#include <../src/Hash.h> |
|
||||||
#endif |
|
||||||
#endif |
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
#define DEFAULT_MAX_SSE_CLIENTS 8 |
|
||||||
#else |
|
||||||
#define DEFAULT_MAX_SSE_CLIENTS 4 |
|
||||||
#endif |
|
||||||
|
|
||||||
class AsyncEventSource; |
|
||||||
class AsyncEventSourceResponse; |
|
||||||
class AsyncEventSourceClient; |
|
||||||
typedef std::function<void(AsyncEventSourceClient *client)> ArEventHandlerFunction; |
|
||||||
|
|
||||||
class AsyncEventSourceMessage { |
|
||||||
private: |
|
||||||
uint8_t * _data;
|
|
||||||
size_t _len; |
|
||||||
size_t _sent; |
|
||||||
//size_t _ack;
|
|
||||||
size_t _acked;
|
|
||||||
public: |
|
||||||
AsyncEventSourceMessage(const char * data, size_t len); |
|
||||||
~AsyncEventSourceMessage(); |
|
||||||
size_t ack(size_t len, uint32_t time __attribute__((unused))); |
|
||||||
size_t send(AsyncClient *client); |
|
||||||
bool finished(){ return _acked == _len; } |
|
||||||
bool sent() { return _sent == _len; } |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncEventSourceClient { |
|
||||||
private: |
|
||||||
AsyncClient *_client; |
|
||||||
AsyncEventSource *_server; |
|
||||||
uint32_t _lastId; |
|
||||||
LinkedList<AsyncEventSourceMessage *> _messageQueue; |
|
||||||
void _queueMessage(AsyncEventSourceMessage *dataMessage); |
|
||||||
void _runQueue(); |
|
||||||
|
|
||||||
public: |
|
||||||
|
|
||||||
AsyncEventSourceClient(AsyncWebServerRequest *request, AsyncEventSource *server); |
|
||||||
~AsyncEventSourceClient(); |
|
||||||
|
|
||||||
AsyncClient* client(){ return _client; } |
|
||||||
void close(); |
|
||||||
void write(const char * message, size_t len); |
|
||||||
void send(const char *message, const char *event=NULL, uint32_t id=0, uint32_t reconnect=0); |
|
||||||
bool connected() const { return (_client != NULL) && _client->connected(); } |
|
||||||
uint32_t lastId() const { return _lastId; } |
|
||||||
size_t packetsWaiting() const { return _messageQueue.length(); } |
|
||||||
|
|
||||||
//system callbacks (do not call)
|
|
||||||
void _onAck(size_t len, uint32_t time); |
|
||||||
void _onPoll();
|
|
||||||
void _onTimeout(uint32_t time); |
|
||||||
void _onDisconnect(); |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncEventSource: public AsyncWebHandler { |
|
||||||
private: |
|
||||||
String _url; |
|
||||||
LinkedList<AsyncEventSourceClient *> _clients; |
|
||||||
ArEventHandlerFunction _connectcb; |
|
||||||
public: |
|
||||||
AsyncEventSource(const String& url); |
|
||||||
~AsyncEventSource(); |
|
||||||
|
|
||||||
const char * url() const { return _url.c_str(); } |
|
||||||
void close(); |
|
||||||
void onConnect(ArEventHandlerFunction cb); |
|
||||||
void send(const char *message, const char *event=NULL, uint32_t id=0, uint32_t reconnect=0); |
|
||||||
size_t count() const; //number clinets connected
|
|
||||||
size_t avgPacketsWaiting() const; |
|
||||||
|
|
||||||
//system callbacks (do not call)
|
|
||||||
void _addClient(AsyncEventSourceClient * client); |
|
||||||
void _handleDisconnect(AsyncEventSourceClient * client); |
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request) override final; |
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request) override final; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncEventSourceResponse: public AsyncWebServerResponse { |
|
||||||
private: |
|
||||||
String _content; |
|
||||||
AsyncEventSource *_server; |
|
||||||
public: |
|
||||||
AsyncEventSourceResponse(AsyncEventSource *server); |
|
||||||
void _respond(AsyncWebServerRequest *request); |
|
||||||
size_t _ack(AsyncWebServerRequest *request, size_t len, uint32_t time); |
|
||||||
bool _sourceValid() const { return true; } |
|
||||||
}; |
|
||||||
|
|
||||||
|
|
||||||
#endif /* ASYNCEVENTSOURCE_H_ */ |
|
@ -1,254 +0,0 @@ |
|||||||
// AsyncJson.h
|
|
||||||
/*
|
|
||||||
Async Response to use with ArduinoJson and AsyncWebServer |
|
||||||
Written by Andrew Melvin (SticilFace) with help from me-no-dev and BBlanchon. |
|
||||||
|
|
||||||
Example of callback in use |
|
||||||
|
|
||||||
server.on("/json", HTTP_ANY, [](AsyncWebServerRequest * request) { |
|
||||||
|
|
||||||
AsyncJsonResponse * response = new AsyncJsonResponse(); |
|
||||||
JsonObject& root = response->getRoot(); |
|
||||||
root["key1"] = "key number one"; |
|
||||||
JsonObject& nested = root.createNestedObject("nested"); |
|
||||||
nested["key1"] = "key number one"; |
|
||||||
|
|
||||||
response->setLength(); |
|
||||||
request->send(response); |
|
||||||
}); |
|
||||||
|
|
||||||
-------------------- |
|
||||||
|
|
||||||
Async Request to use with ArduinoJson and AsyncWebServer |
|
||||||
Written by Arsène von Wyss (avonwyss) |
|
||||||
|
|
||||||
Example |
|
||||||
|
|
||||||
AsyncCallbackJsonWebHandler* handler = new AsyncCallbackJsonWebHandler("/rest/endpoint"); |
|
||||||
handler->onRequest([](AsyncWebServerRequest *request, JsonVariant &json) { |
|
||||||
JsonObject& jsonObj = json.as<JsonObject>(); |
|
||||||
// ...
|
|
||||||
}); |
|
||||||
server.addHandler(handler); |
|
||||||
|
|
||||||
*/ |
|
||||||
#ifndef ASYNC_JSON_H_ |
|
||||||
#define ASYNC_JSON_H_ |
|
||||||
#include <ArduinoJson.h> |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
#include <Print.h> |
|
||||||
|
|
||||||
#if ARDUINOJSON_VERSION_MAJOR == 5 |
|
||||||
#define ARDUINOJSON_5_COMPATIBILITY |
|
||||||
#else |
|
||||||
#ifndef DYNAMIC_JSON_DOCUMENT_SIZE |
|
||||||
#define DYNAMIC_JSON_DOCUMENT_SIZE 1024 |
|
||||||
#endif |
|
||||||
#endif |
|
||||||
|
|
||||||
constexpr const char* JSON_MIMETYPE = "application/json"; |
|
||||||
|
|
||||||
/*
|
|
||||||
* Json Response |
|
||||||
* */ |
|
||||||
|
|
||||||
class ChunkPrint : public Print { |
|
||||||
private: |
|
||||||
uint8_t* _destination; |
|
||||||
size_t _to_skip; |
|
||||||
size_t _to_write; |
|
||||||
size_t _pos; |
|
||||||
public: |
|
||||||
ChunkPrint(uint8_t* destination, size_t from, size_t len) |
|
||||||
: _destination(destination), _to_skip(from), _to_write(len), _pos{0} {} |
|
||||||
virtual ~ChunkPrint(){} |
|
||||||
size_t write(uint8_t c){ |
|
||||||
if (_to_skip > 0) { |
|
||||||
_to_skip--; |
|
||||||
return 1; |
|
||||||
} else if (_to_write > 0) { |
|
||||||
_to_write--; |
|
||||||
_destination[_pos++] = c; |
|
||||||
return 1; |
|
||||||
} |
|
||||||
return 0; |
|
||||||
} |
|
||||||
size_t write(const uint8_t *buffer, size_t size) |
|
||||||
{ |
|
||||||
return this->Print::write(buffer, size); |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncJsonResponse: public AsyncAbstractResponse { |
|
||||||
protected: |
|
||||||
|
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
DynamicJsonBuffer _jsonBuffer; |
|
||||||
#else |
|
||||||
DynamicJsonDocument _jsonBuffer; |
|
||||||
#endif |
|
||||||
|
|
||||||
JsonVariant _root; |
|
||||||
bool _isValid; |
|
||||||
|
|
||||||
public:
|
|
||||||
|
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
AsyncJsonResponse(bool isArray=false): _isValid{false} { |
|
||||||
_code = 200; |
|
||||||
_contentType = JSON_MIMETYPE; |
|
||||||
if(isArray) |
|
||||||
_root = _jsonBuffer.createArray(); |
|
||||||
else |
|
||||||
_root = _jsonBuffer.createObject(); |
|
||||||
} |
|
||||||
#else |
|
||||||
AsyncJsonResponse(bool isArray=false, size_t maxJsonBufferSize = DYNAMIC_JSON_DOCUMENT_SIZE) : _jsonBuffer(maxJsonBufferSize), _isValid{false} { |
|
||||||
_code = 200; |
|
||||||
_contentType = JSON_MIMETYPE; |
|
||||||
if(isArray) |
|
||||||
_root = _jsonBuffer.createNestedArray(); |
|
||||||
else |
|
||||||
_root = _jsonBuffer.createNestedObject(); |
|
||||||
} |
|
||||||
#endif |
|
||||||
|
|
||||||
~AsyncJsonResponse() {} |
|
||||||
JsonVariant & getRoot() { return _root; } |
|
||||||
bool _sourceValid() const { return _isValid; } |
|
||||||
size_t setLength() { |
|
||||||
|
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
_contentLength = _root.measureLength(); |
|
||||||
#else |
|
||||||
_contentLength = measureJson(_root); |
|
||||||
#endif |
|
||||||
|
|
||||||
if (_contentLength) { _isValid = true; } |
|
||||||
return _contentLength; |
|
||||||
} |
|
||||||
|
|
||||||
size_t getSize() { return _jsonBuffer.size(); } |
|
||||||
|
|
||||||
size_t _fillBuffer(uint8_t *data, size_t len){ |
|
||||||
ChunkPrint dest(data, _sentLength, len); |
|
||||||
|
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
_root.printTo( dest ) ; |
|
||||||
#else |
|
||||||
serializeJson(_root, dest); |
|
||||||
#endif |
|
||||||
return len; |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
class PrettyAsyncJsonResponse: public AsyncJsonResponse {
|
|
||||||
public: |
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
PrettyAsyncJsonResponse (bool isArray=false) : AsyncJsonResponse{isArray} {} |
|
||||||
#else |
|
||||||
PrettyAsyncJsonResponse (bool isArray=false, size_t maxJsonBufferSize = DYNAMIC_JSON_DOCUMENT_SIZE) : AsyncJsonResponse{isArray, maxJsonBufferSize} {} |
|
||||||
#endif |
|
||||||
size_t setLength () { |
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
_contentLength = _root.measurePrettyLength (); |
|
||||||
#else |
|
||||||
_contentLength = measureJsonPretty(_root); |
|
||||||
#endif |
|
||||||
if (_contentLength) {_isValid = true;} |
|
||||||
return _contentLength; |
|
||||||
} |
|
||||||
size_t _fillBuffer (uint8_t *data, size_t len) { |
|
||||||
ChunkPrint dest (data, _sentLength, len); |
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
_root.prettyPrintTo (dest); |
|
||||||
#else |
|
||||||
serializeJsonPretty(_root, dest); |
|
||||||
#endif |
|
||||||
return len; |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
typedef std::function<void(AsyncWebServerRequest *request, JsonVariant &json)> ArJsonRequestHandlerFunction; |
|
||||||
|
|
||||||
class AsyncCallbackJsonWebHandler: public AsyncWebHandler { |
|
||||||
private: |
|
||||||
protected: |
|
||||||
const String _uri; |
|
||||||
WebRequestMethodComposite _method; |
|
||||||
ArJsonRequestHandlerFunction _onRequest; |
|
||||||
size_t _contentLength; |
|
||||||
#ifndef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
const size_t maxJsonBufferSize; |
|
||||||
#endif |
|
||||||
size_t _maxContentLength; |
|
||||||
public: |
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
AsyncCallbackJsonWebHandler(const String& uri, ArJsonRequestHandlerFunction onRequest)
|
|
||||||
: _uri(uri), _method(HTTP_POST|HTTP_PUT|HTTP_PATCH), _onRequest(onRequest), _maxContentLength(16384) {} |
|
||||||
#else |
|
||||||
AsyncCallbackJsonWebHandler(const String& uri, ArJsonRequestHandlerFunction onRequest, size_t maxJsonBufferSize=DYNAMIC_JSON_DOCUMENT_SIZE)
|
|
||||||
: _uri(uri), _method(HTTP_POST|HTTP_PUT|HTTP_PATCH), _onRequest(onRequest), maxJsonBufferSize(maxJsonBufferSize), _maxContentLength(16384) {} |
|
||||||
#endif |
|
||||||
|
|
||||||
void setMethod(WebRequestMethodComposite method){ _method = method; } |
|
||||||
void setMaxContentLength(int maxContentLength){ _maxContentLength = maxContentLength; } |
|
||||||
void onRequest(ArJsonRequestHandlerFunction fn){ _onRequest = fn; } |
|
||||||
|
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request) override final{ |
|
||||||
if(!_onRequest) |
|
||||||
return false; |
|
||||||
|
|
||||||
if(!(_method & request->method())) |
|
||||||
return false; |
|
||||||
|
|
||||||
if(_uri.length() && (_uri != request->url() && !request->url().startsWith(_uri+"/"))) |
|
||||||
return false; |
|
||||||
|
|
||||||
if ( !request->contentType().equalsIgnoreCase(JSON_MIMETYPE) ) |
|
||||||
return false; |
|
||||||
|
|
||||||
request->addInterestingHeader("ANY"); |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request) override final { |
|
||||||
if(_onRequest) { |
|
||||||
if (request->_tempObject != NULL) { |
|
||||||
|
|
||||||
#ifdef ARDUINOJSON_5_COMPATIBILITY |
|
||||||
DynamicJsonBuffer jsonBuffer; |
|
||||||
JsonVariant json = jsonBuffer.parse((uint8_t*)(request->_tempObject)); |
|
||||||
if (json.success()) { |
|
||||||
#else |
|
||||||
DynamicJsonDocument jsonBuffer(this->maxJsonBufferSize); |
|
||||||
DeserializationError error = deserializeJson(jsonBuffer, (uint8_t*)(request->_tempObject)); |
|
||||||
if(!error) { |
|
||||||
JsonVariant json = jsonBuffer.as<JsonVariant>(); |
|
||||||
#endif |
|
||||||
|
|
||||||
_onRequest(request, json); |
|
||||||
return; |
|
||||||
} |
|
||||||
} |
|
||||||
request->send(_contentLength > _maxContentLength ? 413 : 400); |
|
||||||
} else { |
|
||||||
request->send(500); |
|
||||||
} |
|
||||||
} |
|
||||||
virtual void handleUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) override final { |
|
||||||
} |
|
||||||
virtual void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total) override final { |
|
||||||
if (_onRequest) { |
|
||||||
_contentLength = total; |
|
||||||
if (total > 0 && request->_tempObject == NULL && total < _maxContentLength) { |
|
||||||
request->_tempObject = malloc(total); |
|
||||||
} |
|
||||||
if (request->_tempObject != NULL) { |
|
||||||
memcpy((uint8_t*)(request->_tempObject) + index, data, len); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
virtual bool isRequestHandlerTrivial() override final {return _onRequest ? false : true;} |
|
||||||
}; |
|
||||||
#endif |
|
File diff suppressed because it is too large
Load Diff
@ -1,350 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#ifndef ASYNCWEBSOCKET_H_ |
|
||||||
#define ASYNCWEBSOCKET_H_ |
|
||||||
|
|
||||||
#include <Arduino.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#define WS_MAX_QUEUED_MESSAGES 32 |
|
||||||
#else |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#define WS_MAX_QUEUED_MESSAGES 8 |
|
||||||
#endif |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
|
|
||||||
#include "AsyncWebSynchronization.h" |
|
||||||
|
|
||||||
#ifdef ESP8266 |
|
||||||
#include <Hash.h> |
|
||||||
#ifdef CRYPTO_HASH_h // include Hash.h from espressif framework if the first include was from the crypto library
|
|
||||||
#include <../src/Hash.h> |
|
||||||
#endif |
|
||||||
#endif |
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
#define DEFAULT_MAX_WS_CLIENTS 8 |
|
||||||
#else |
|
||||||
#define DEFAULT_MAX_WS_CLIENTS 4 |
|
||||||
#endif |
|
||||||
|
|
||||||
class AsyncWebSocket; |
|
||||||
class AsyncWebSocketResponse; |
|
||||||
class AsyncWebSocketClient; |
|
||||||
class AsyncWebSocketControl; |
|
||||||
|
|
||||||
typedef struct { |
|
||||||
/** Message type as defined by enum AwsFrameType.
|
|
||||||
* Note: Applications will only see WS_TEXT and WS_BINARY. |
|
||||||
* All other types are handled by the library. */ |
|
||||||
uint8_t message_opcode; |
|
||||||
/** Frame number of a fragmented message. */ |
|
||||||
uint32_t num; |
|
||||||
/** Is this the last frame in a fragmented message ?*/ |
|
||||||
uint8_t final; |
|
||||||
/** Is this frame masked? */ |
|
||||||
uint8_t masked; |
|
||||||
/** Message type as defined by enum AwsFrameType.
|
|
||||||
* This value is the same as message_opcode for non-fragmented |
|
||||||
* messages, but may also be WS_CONTINUATION in a fragmented message. */ |
|
||||||
uint8_t opcode; |
|
||||||
/** Length of the current frame.
|
|
||||||
* This equals the total length of the message if num == 0 && final == true */ |
|
||||||
uint64_t len; |
|
||||||
/** Mask key */ |
|
||||||
uint8_t mask[4]; |
|
||||||
/** Offset of the data inside the current frame. */ |
|
||||||
uint64_t index; |
|
||||||
} AwsFrameInfo; |
|
||||||
|
|
||||||
typedef enum { WS_DISCONNECTED, WS_CONNECTED, WS_DISCONNECTING } AwsClientStatus; |
|
||||||
typedef enum { WS_CONTINUATION, WS_TEXT, WS_BINARY, WS_DISCONNECT = 0x08, WS_PING, WS_PONG } AwsFrameType; |
|
||||||
typedef enum { WS_MSG_SENDING, WS_MSG_SENT, WS_MSG_ERROR } AwsMessageStatus; |
|
||||||
typedef enum { WS_EVT_CONNECT, WS_EVT_DISCONNECT, WS_EVT_PONG, WS_EVT_ERROR, WS_EVT_DATA } AwsEventType; |
|
||||||
|
|
||||||
class AsyncWebSocketMessageBuffer { |
|
||||||
private: |
|
||||||
uint8_t * _data; |
|
||||||
size_t _len; |
|
||||||
bool _lock;
|
|
||||||
uint32_t _count;
|
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebSocketMessageBuffer(); |
|
||||||
AsyncWebSocketMessageBuffer(size_t size); |
|
||||||
AsyncWebSocketMessageBuffer(uint8_t * data, size_t size);
|
|
||||||
AsyncWebSocketMessageBuffer(const AsyncWebSocketMessageBuffer &);
|
|
||||||
AsyncWebSocketMessageBuffer(AsyncWebSocketMessageBuffer &&);
|
|
||||||
~AsyncWebSocketMessageBuffer();
|
|
||||||
void operator ++(int i) { (void)i; _count++; } |
|
||||||
void operator --(int i) { (void)i; if (_count > 0) { _count--; } ; } |
|
||||||
bool reserve(size_t size); |
|
||||||
void lock() { _lock = true; } |
|
||||||
void unlock() { _lock = false; } |
|
||||||
uint8_t * get() { return _data; } |
|
||||||
size_t length() { return _len; } |
|
||||||
uint32_t count() { return _count; } |
|
||||||
bool canDelete() { return (!_count && !_lock); }
|
|
||||||
|
|
||||||
friend AsyncWebSocket;
|
|
||||||
|
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncWebSocketMessage { |
|
||||||
protected: |
|
||||||
uint8_t _opcode; |
|
||||||
bool _mask; |
|
||||||
AwsMessageStatus _status; |
|
||||||
public: |
|
||||||
AsyncWebSocketMessage():_opcode(WS_TEXT),_mask(false),_status(WS_MSG_ERROR){} |
|
||||||
virtual ~AsyncWebSocketMessage(){} |
|
||||||
virtual void ack(size_t len __attribute__((unused)), uint32_t time __attribute__((unused))){} |
|
||||||
virtual size_t send(AsyncClient *client __attribute__((unused))){ return 0; } |
|
||||||
virtual bool finished(){ return _status != WS_MSG_SENDING; } |
|
||||||
virtual bool betweenFrames() const { return false; } |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncWebSocketBasicMessage: public AsyncWebSocketMessage { |
|
||||||
private: |
|
||||||
size_t _len; |
|
||||||
size_t _sent; |
|
||||||
size_t _ack; |
|
||||||
size_t _acked; |
|
||||||
uint8_t * _data; |
|
||||||
public: |
|
||||||
AsyncWebSocketBasicMessage(const char * data, size_t len, uint8_t opcode=WS_TEXT, bool mask=false); |
|
||||||
AsyncWebSocketBasicMessage(uint8_t opcode=WS_TEXT, bool mask=false); |
|
||||||
virtual ~AsyncWebSocketBasicMessage() override; |
|
||||||
virtual bool betweenFrames() const override { return _acked == _ack; } |
|
||||||
virtual void ack(size_t len, uint32_t time) override ; |
|
||||||
virtual size_t send(AsyncClient *client) override ; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncWebSocketMultiMessage: public AsyncWebSocketMessage { |
|
||||||
private: |
|
||||||
uint8_t * _data; |
|
||||||
size_t _len; |
|
||||||
size_t _sent; |
|
||||||
size_t _ack; |
|
||||||
size_t _acked; |
|
||||||
AsyncWebSocketMessageBuffer * _WSbuffer;
|
|
||||||
public: |
|
||||||
AsyncWebSocketMultiMessage(AsyncWebSocketMessageBuffer * buffer, uint8_t opcode=WS_TEXT, bool mask=false);
|
|
||||||
virtual ~AsyncWebSocketMultiMessage() override; |
|
||||||
virtual bool betweenFrames() const override { return _acked == _ack; } |
|
||||||
virtual void ack(size_t len, uint32_t time) override ; |
|
||||||
virtual size_t send(AsyncClient *client) override ; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncWebSocketClient { |
|
||||||
private: |
|
||||||
AsyncClient *_client; |
|
||||||
AsyncWebSocket *_server; |
|
||||||
uint32_t _clientId; |
|
||||||
AwsClientStatus _status; |
|
||||||
|
|
||||||
LinkedList<AsyncWebSocketControl *> _controlQueue; |
|
||||||
LinkedList<AsyncWebSocketMessage *> _messageQueue; |
|
||||||
|
|
||||||
uint8_t _pstate; |
|
||||||
AwsFrameInfo _pinfo; |
|
||||||
|
|
||||||
uint32_t _lastMessageTime; |
|
||||||
uint32_t _keepAlivePeriod; |
|
||||||
|
|
||||||
void _queueMessage(AsyncWebSocketMessage *dataMessage); |
|
||||||
void _queueControl(AsyncWebSocketControl *controlMessage); |
|
||||||
void _runQueue(); |
|
||||||
|
|
||||||
public: |
|
||||||
void *_tempObject; |
|
||||||
|
|
||||||
AsyncWebSocketClient(AsyncWebServerRequest *request, AsyncWebSocket *server); |
|
||||||
~AsyncWebSocketClient(); |
|
||||||
|
|
||||||
//client id increments for the given server
|
|
||||||
uint32_t id(){ return _clientId; } |
|
||||||
AwsClientStatus status(){ return _status; } |
|
||||||
AsyncClient* client(){ return _client; } |
|
||||||
AsyncWebSocket *server(){ return _server; } |
|
||||||
AwsFrameInfo const &pinfo() const { return _pinfo; } |
|
||||||
|
|
||||||
IPAddress remoteIP(); |
|
||||||
uint16_t remotePort(); |
|
||||||
|
|
||||||
//control frames
|
|
||||||
void close(uint16_t code=0, const char * message=NULL); |
|
||||||
void ping(uint8_t *data=NULL, size_t len=0); |
|
||||||
|
|
||||||
//set auto-ping period in seconds. disabled if zero (default)
|
|
||||||
void keepAlivePeriod(uint16_t seconds){ |
|
||||||
_keepAlivePeriod = seconds * 1000; |
|
||||||
} |
|
||||||
uint16_t keepAlivePeriod(){ |
|
||||||
return (uint16_t)(_keepAlivePeriod / 1000); |
|
||||||
} |
|
||||||
|
|
||||||
//data packets
|
|
||||||
void message(AsyncWebSocketMessage *message){ _queueMessage(message); } |
|
||||||
bool queueIsFull(); |
|
||||||
|
|
||||||
size_t printf(const char *format, ...) __attribute__ ((format (printf, 2, 3))); |
|
||||||
#ifndef ESP32 |
|
||||||
size_t printf_P(PGM_P formatP, ...) __attribute__ ((format (printf, 2, 3))); |
|
||||||
#endif |
|
||||||
void text(const char * message, size_t len); |
|
||||||
void text(const char * message); |
|
||||||
void text(uint8_t * message, size_t len); |
|
||||||
void text(char * message); |
|
||||||
void text(const String &message); |
|
||||||
void text(const __FlashStringHelper *data); |
|
||||||
void text(AsyncWebSocketMessageBuffer *buffer);
|
|
||||||
|
|
||||||
void binary(const char * message, size_t len); |
|
||||||
void binary(const char * message); |
|
||||||
void binary(uint8_t * message, size_t len); |
|
||||||
void binary(char * message); |
|
||||||
void binary(const String &message); |
|
||||||
void binary(const __FlashStringHelper *data, size_t len); |
|
||||||
void binary(AsyncWebSocketMessageBuffer *buffer);
|
|
||||||
|
|
||||||
bool canSend() { return _messageQueue.length() < WS_MAX_QUEUED_MESSAGES; } |
|
||||||
|
|
||||||
//system callbacks (do not call)
|
|
||||||
void _onAck(size_t len, uint32_t time); |
|
||||||
void _onError(int8_t); |
|
||||||
void _onPoll(); |
|
||||||
void _onTimeout(uint32_t time); |
|
||||||
void _onDisconnect(); |
|
||||||
void _onData(void *pbuf, size_t plen); |
|
||||||
}; |
|
||||||
|
|
||||||
typedef std::function<void(AsyncWebSocket * server, AsyncWebSocketClient * client, AwsEventType type, void * arg, uint8_t *data, size_t len)> AwsEventHandler; |
|
||||||
|
|
||||||
//WebServer Handler implementation that plays the role of a socket server
|
|
||||||
class AsyncWebSocket: public AsyncWebHandler { |
|
||||||
public: |
|
||||||
typedef LinkedList<AsyncWebSocketClient *> AsyncWebSocketClientLinkedList; |
|
||||||
private: |
|
||||||
String _url; |
|
||||||
AsyncWebSocketClientLinkedList _clients; |
|
||||||
uint32_t _cNextId; |
|
||||||
AwsEventHandler _eventHandler; |
|
||||||
bool _enabled; |
|
||||||
AsyncWebLock _lock; |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebSocket(const String& url); |
|
||||||
~AsyncWebSocket(); |
|
||||||
const char * url() const { return _url.c_str(); } |
|
||||||
void enable(bool e){ _enabled = e; } |
|
||||||
bool enabled() const { return _enabled; } |
|
||||||
bool availableForWriteAll(); |
|
||||||
bool availableForWrite(uint32_t id); |
|
||||||
|
|
||||||
size_t count() const; |
|
||||||
AsyncWebSocketClient * client(uint32_t id); |
|
||||||
bool hasClient(uint32_t id){ return client(id) != NULL; } |
|
||||||
|
|
||||||
void close(uint32_t id, uint16_t code=0, const char * message=NULL); |
|
||||||
void closeAll(uint16_t code=0, const char * message=NULL); |
|
||||||
void cleanupClients(uint16_t maxClients = DEFAULT_MAX_WS_CLIENTS); |
|
||||||
|
|
||||||
void ping(uint32_t id, uint8_t *data=NULL, size_t len=0); |
|
||||||
void pingAll(uint8_t *data=NULL, size_t len=0); // done
|
|
||||||
|
|
||||||
void text(uint32_t id, const char * message, size_t len); |
|
||||||
void text(uint32_t id, const char * message); |
|
||||||
void text(uint32_t id, uint8_t * message, size_t len); |
|
||||||
void text(uint32_t id, char * message); |
|
||||||
void text(uint32_t id, const String &message); |
|
||||||
void text(uint32_t id, const __FlashStringHelper *message); |
|
||||||
|
|
||||||
void textAll(const char * message, size_t len); |
|
||||||
void textAll(const char * message); |
|
||||||
void textAll(uint8_t * message, size_t len); |
|
||||||
void textAll(char * message); |
|
||||||
void textAll(const String &message); |
|
||||||
void textAll(const __FlashStringHelper *message); // need to convert
|
|
||||||
void textAll(AsyncWebSocketMessageBuffer * buffer);
|
|
||||||
|
|
||||||
void binary(uint32_t id, const char * message, size_t len); |
|
||||||
void binary(uint32_t id, const char * message); |
|
||||||
void binary(uint32_t id, uint8_t * message, size_t len); |
|
||||||
void binary(uint32_t id, char * message); |
|
||||||
void binary(uint32_t id, const String &message); |
|
||||||
void binary(uint32_t id, const __FlashStringHelper *message, size_t len); |
|
||||||
|
|
||||||
void binaryAll(const char * message, size_t len); |
|
||||||
void binaryAll(const char * message); |
|
||||||
void binaryAll(uint8_t * message, size_t len); |
|
||||||
void binaryAll(char * message); |
|
||||||
void binaryAll(const String &message); |
|
||||||
void binaryAll(const __FlashStringHelper *message, size_t len); |
|
||||||
void binaryAll(AsyncWebSocketMessageBuffer * buffer);
|
|
||||||
|
|
||||||
void message(uint32_t id, AsyncWebSocketMessage *message); |
|
||||||
void messageAll(AsyncWebSocketMultiMessage *message); |
|
||||||
|
|
||||||
size_t printf(uint32_t id, const char *format, ...) __attribute__ ((format (printf, 3, 4))); |
|
||||||
size_t printfAll(const char *format, ...) __attribute__ ((format (printf, 2, 3))); |
|
||||||
#ifndef ESP32 |
|
||||||
size_t printf_P(uint32_t id, PGM_P formatP, ...) __attribute__ ((format (printf, 3, 4))); |
|
||||||
#endif |
|
||||||
size_t printfAll_P(PGM_P formatP, ...) __attribute__ ((format (printf, 2, 3))); |
|
||||||
|
|
||||||
//event listener
|
|
||||||
void onEvent(AwsEventHandler handler){ |
|
||||||
_eventHandler = handler; |
|
||||||
} |
|
||||||
|
|
||||||
//system callbacks (do not call)
|
|
||||||
uint32_t _getNextId(){ return _cNextId++; } |
|
||||||
void _addClient(AsyncWebSocketClient * client); |
|
||||||
void _handleDisconnect(AsyncWebSocketClient * client); |
|
||||||
void _handleEvent(AsyncWebSocketClient * client, AwsEventType type, void * arg, uint8_t *data, size_t len); |
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request) override final; |
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request) override final; |
|
||||||
|
|
||||||
|
|
||||||
// messagebuffer functions/objects.
|
|
||||||
AsyncWebSocketMessageBuffer * makeBuffer(size_t size = 0);
|
|
||||||
AsyncWebSocketMessageBuffer * makeBuffer(uint8_t * data, size_t size);
|
|
||||||
LinkedList<AsyncWebSocketMessageBuffer *> _buffers; |
|
||||||
void _cleanBuffers();
|
|
||||||
|
|
||||||
AsyncWebSocketClientLinkedList getClients() const; |
|
||||||
}; |
|
||||||
|
|
||||||
//WebServer response to authenticate the socket and detach the tcp client from the web server request
|
|
||||||
class AsyncWebSocketResponse: public AsyncWebServerResponse { |
|
||||||
private: |
|
||||||
String _content; |
|
||||||
AsyncWebSocket *_server; |
|
||||||
public: |
|
||||||
AsyncWebSocketResponse(const String& key, AsyncWebSocket *server); |
|
||||||
void _respond(AsyncWebServerRequest *request); |
|
||||||
size_t _ack(AsyncWebServerRequest *request, size_t len, uint32_t time); |
|
||||||
bool _sourceValid() const { return true; } |
|
||||||
}; |
|
||||||
|
|
||||||
|
|
||||||
#endif /* ASYNCWEBSOCKET_H_ */ |
|
@ -1,87 +0,0 @@ |
|||||||
#ifndef ASYNCWEBSYNCHRONIZATION_H_ |
|
||||||
#define ASYNCWEBSYNCHRONIZATION_H_ |
|
||||||
|
|
||||||
// Synchronisation is only available on ESP32, as the ESP8266 isn't using FreeRTOS by default
|
|
||||||
|
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
|
|
||||||
// This is the ESP32 version of the Sync Lock, using the FreeRTOS Semaphore
|
|
||||||
class AsyncWebLock |
|
||||||
{ |
|
||||||
private: |
|
||||||
SemaphoreHandle_t _lock; |
|
||||||
mutable void *_lockedBy; |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebLock() { |
|
||||||
_lock = xSemaphoreCreateBinary(); |
|
||||||
_lockedBy = NULL; |
|
||||||
xSemaphoreGive(_lock); |
|
||||||
} |
|
||||||
|
|
||||||
~AsyncWebLock() { |
|
||||||
vSemaphoreDelete(_lock); |
|
||||||
} |
|
||||||
|
|
||||||
bool lock() const { |
|
||||||
extern void *pxCurrentTCB; |
|
||||||
if (_lockedBy != pxCurrentTCB) { |
|
||||||
xSemaphoreTake(_lock, portMAX_DELAY); |
|
||||||
_lockedBy = pxCurrentTCB; |
|
||||||
return true; |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
void unlock() const { |
|
||||||
_lockedBy = NULL; |
|
||||||
xSemaphoreGive(_lock); |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
#else |
|
||||||
|
|
||||||
// This is the 8266 version of the Sync Lock which is currently unimplemented
|
|
||||||
class AsyncWebLock |
|
||||||
{ |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebLock() { |
|
||||||
} |
|
||||||
|
|
||||||
~AsyncWebLock() { |
|
||||||
} |
|
||||||
|
|
||||||
bool lock() const { |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
void unlock() const { |
|
||||||
} |
|
||||||
}; |
|
||||||
#endif |
|
||||||
|
|
||||||
class AsyncWebLockGuard |
|
||||||
{ |
|
||||||
private: |
|
||||||
const AsyncWebLock *_lock; |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebLockGuard(const AsyncWebLock &l) { |
|
||||||
if (l.lock()) { |
|
||||||
_lock = &l; |
|
||||||
} else { |
|
||||||
_lock = NULL; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
~AsyncWebLockGuard() { |
|
||||||
if (_lock) { |
|
||||||
_lock->unlock(); |
|
||||||
} |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
#endif // ASYNCWEBSYNCHRONIZATION_H_
|
|
@ -1,471 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#ifndef _ESPAsyncWebServer_H_ |
|
||||||
#define _ESPAsyncWebServer_H_ |
|
||||||
|
|
||||||
#include "Arduino.h" |
|
||||||
|
|
||||||
#include <functional> |
|
||||||
#include "FS.h" |
|
||||||
|
|
||||||
#include "StringArray.h" |
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
#include <WiFi.h> |
|
||||||
#include <AsyncTCP.h> |
|
||||||
#elif defined(ESP8266) |
|
||||||
#include <ESP8266WiFi.h> |
|
||||||
#include <ESPAsyncTCP.h> |
|
||||||
#else |
|
||||||
#error Platform not supported |
|
||||||
#endif |
|
||||||
|
|
||||||
#ifdef ASYNCWEBSERVER_REGEX |
|
||||||
#define ASYNCWEBSERVER_REGEX_ATTRIBUTE |
|
||||||
#else |
|
||||||
#define ASYNCWEBSERVER_REGEX_ATTRIBUTE __attribute__((warning("ASYNCWEBSERVER_REGEX not defined"))) |
|
||||||
#endif |
|
||||||
|
|
||||||
#define DEBUGF(...) //Serial.printf(__VA_ARGS__)
|
|
||||||
|
|
||||||
class AsyncWebServer; |
|
||||||
class AsyncWebServerRequest; |
|
||||||
class AsyncWebServerResponse; |
|
||||||
class AsyncWebHeader; |
|
||||||
class AsyncWebParameter; |
|
||||||
class AsyncWebRewrite; |
|
||||||
class AsyncWebHandler; |
|
||||||
class AsyncStaticWebHandler; |
|
||||||
class AsyncCallbackWebHandler; |
|
||||||
class AsyncResponseStream; |
|
||||||
|
|
||||||
#ifndef WEBSERVER_H |
|
||||||
typedef enum { |
|
||||||
HTTP_GET = 0b00000001, |
|
||||||
HTTP_POST = 0b00000010, |
|
||||||
HTTP_DELETE = 0b00000100, |
|
||||||
HTTP_PUT = 0b00001000, |
|
||||||
HTTP_PATCH = 0b00010000, |
|
||||||
HTTP_HEAD = 0b00100000, |
|
||||||
HTTP_OPTIONS = 0b01000000, |
|
||||||
HTTP_ANY = 0b01111111, |
|
||||||
} WebRequestMethod; |
|
||||||
#endif |
|
||||||
|
|
||||||
//if this value is returned when asked for data, packet will not be sent and you will be asked for data again
|
|
||||||
#define RESPONSE_TRY_AGAIN 0xFFFFFFFF |
|
||||||
|
|
||||||
typedef uint8_t WebRequestMethodComposite; |
|
||||||
typedef std::function<void(void)> ArDisconnectHandler; |
|
||||||
|
|
||||||
/*
|
|
||||||
* PARAMETER :: Chainable object to hold GET/POST and FILE parameters |
|
||||||
* */ |
|
||||||
|
|
||||||
class AsyncWebParameter { |
|
||||||
private: |
|
||||||
String _name; |
|
||||||
String _value; |
|
||||||
size_t _size; |
|
||||||
bool _isForm; |
|
||||||
bool _isFile; |
|
||||||
|
|
||||||
public: |
|
||||||
|
|
||||||
AsyncWebParameter(const String& name, const String& value, bool form=false, bool file=false, size_t size=0): _name(name), _value(value), _size(size), _isForm(form), _isFile(file){} |
|
||||||
const String& name() const { return _name; } |
|
||||||
const String& value() const { return _value; } |
|
||||||
size_t size() const { return _size; } |
|
||||||
bool isPost() const { return _isForm; } |
|
||||||
bool isFile() const { return _isFile; } |
|
||||||
}; |
|
||||||
|
|
||||||
/*
|
|
||||||
* HEADER :: Chainable object to hold the headers |
|
||||||
* */ |
|
||||||
|
|
||||||
class AsyncWebHeader { |
|
||||||
private: |
|
||||||
String _name; |
|
||||||
String _value; |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebHeader(const String& name, const String& value): _name(name), _value(value){} |
|
||||||
AsyncWebHeader(const String& data): _name(), _value(){ |
|
||||||
if(!data) return; |
|
||||||
int index = data.indexOf(':'); |
|
||||||
if (index < 0) return; |
|
||||||
_name = data.substring(0, index); |
|
||||||
_value = data.substring(index + 2); |
|
||||||
} |
|
||||||
~AsyncWebHeader(){} |
|
||||||
const String& name() const { return _name; } |
|
||||||
const String& value() const { return _value; } |
|
||||||
String toString() const { return String(_name+": "+_value+"\r\n"); } |
|
||||||
}; |
|
||||||
|
|
||||||
/*
|
|
||||||
* REQUEST :: Each incoming Client is wrapped inside a Request and both live together until disconnect |
|
||||||
* */ |
|
||||||
|
|
||||||
typedef enum { RCT_NOT_USED = -1, RCT_DEFAULT = 0, RCT_HTTP, RCT_WS, RCT_EVENT, RCT_MAX } RequestedConnectionType; |
|
||||||
|
|
||||||
typedef std::function<size_t(uint8_t*, size_t, size_t)> AwsResponseFiller; |
|
||||||
typedef std::function<String(const String&)> AwsTemplateProcessor; |
|
||||||
|
|
||||||
class AsyncWebServerRequest { |
|
||||||
using File = fs::File; |
|
||||||
using FS = fs::FS; |
|
||||||
friend class AsyncWebServer; |
|
||||||
friend class AsyncCallbackWebHandler; |
|
||||||
private: |
|
||||||
AsyncClient* _client; |
|
||||||
AsyncWebServer* _server; |
|
||||||
AsyncWebHandler* _handler; |
|
||||||
AsyncWebServerResponse* _response; |
|
||||||
StringArray _interestingHeaders; |
|
||||||
ArDisconnectHandler _onDisconnectfn; |
|
||||||
|
|
||||||
String _temp; |
|
||||||
uint8_t _parseState; |
|
||||||
|
|
||||||
uint8_t _version; |
|
||||||
WebRequestMethodComposite _method; |
|
||||||
String _url; |
|
||||||
String _host; |
|
||||||
String _contentType; |
|
||||||
String _boundary; |
|
||||||
String _authorization; |
|
||||||
RequestedConnectionType _reqconntype; |
|
||||||
void _removeNotInterestingHeaders(); |
|
||||||
bool _isDigest; |
|
||||||
bool _isMultipart; |
|
||||||
bool _isPlainPost; |
|
||||||
bool _expectingContinue; |
|
||||||
size_t _contentLength; |
|
||||||
size_t _parsedLength; |
|
||||||
|
|
||||||
LinkedList<AsyncWebHeader *> _headers; |
|
||||||
LinkedList<AsyncWebParameter *> _params; |
|
||||||
LinkedList<String *> _pathParams; |
|
||||||
|
|
||||||
uint8_t _multiParseState; |
|
||||||
uint8_t _boundaryPosition; |
|
||||||
size_t _itemStartIndex; |
|
||||||
size_t _itemSize; |
|
||||||
String _itemName; |
|
||||||
String _itemFilename; |
|
||||||
String _itemType; |
|
||||||
String _itemValue; |
|
||||||
uint8_t *_itemBuffer; |
|
||||||
size_t _itemBufferIndex; |
|
||||||
bool _itemIsFile; |
|
||||||
|
|
||||||
void _onPoll(); |
|
||||||
void _onAck(size_t len, uint32_t time); |
|
||||||
void _onError(int8_t error); |
|
||||||
void _onTimeout(uint32_t time); |
|
||||||
void _onDisconnect(); |
|
||||||
void _onData(void *buf, size_t len); |
|
||||||
|
|
||||||
void _addParam(AsyncWebParameter*); |
|
||||||
void _addPathParam(const char *param); |
|
||||||
|
|
||||||
bool _parseReqHead(); |
|
||||||
bool _parseReqHeader(); |
|
||||||
void _parseLine(); |
|
||||||
void _parsePlainPostChar(uint8_t data); |
|
||||||
void _parseMultipartPostByte(uint8_t data, bool last); |
|
||||||
void _addGetParams(const String& params); |
|
||||||
|
|
||||||
void _handleUploadStart(); |
|
||||||
void _handleUploadByte(uint8_t data, bool last); |
|
||||||
void _handleUploadEnd(); |
|
||||||
|
|
||||||
public: |
|
||||||
File _tempFile; |
|
||||||
void *_tempObject; |
|
||||||
|
|
||||||
AsyncWebServerRequest(AsyncWebServer*, AsyncClient*); |
|
||||||
~AsyncWebServerRequest(); |
|
||||||
|
|
||||||
AsyncClient* client(){ return _client; } |
|
||||||
uint8_t version() const { return _version; } |
|
||||||
WebRequestMethodComposite method() const { return _method; } |
|
||||||
const String& url() const { return _url; } |
|
||||||
const String& host() const { return _host; } |
|
||||||
const String& contentType() const { return _contentType; } |
|
||||||
size_t contentLength() const { return _contentLength; } |
|
||||||
bool multipart() const { return _isMultipart; } |
|
||||||
const char * methodToString() const; |
|
||||||
const char * requestedConnTypeToString() const; |
|
||||||
RequestedConnectionType requestedConnType() const { return _reqconntype; } |
|
||||||
bool isExpectedRequestedConnType(RequestedConnectionType erct1, RequestedConnectionType erct2 = RCT_NOT_USED, RequestedConnectionType erct3 = RCT_NOT_USED); |
|
||||||
void onDisconnect (ArDisconnectHandler fn); |
|
||||||
|
|
||||||
//hash is the string representation of:
|
|
||||||
// base64(user:pass) for basic or
|
|
||||||
// user:realm:md5(user:realm:pass) for digest
|
|
||||||
bool authenticate(const char * hash); |
|
||||||
bool authenticate(const char * username, const char * password, const char * realm = NULL, bool passwordIsHash = false); |
|
||||||
void requestAuthentication(const char * realm = NULL, bool isDigest = true); |
|
||||||
|
|
||||||
void setHandler(AsyncWebHandler *handler){ _handler = handler; } |
|
||||||
void addInterestingHeader(const String& name); |
|
||||||
|
|
||||||
void redirect(const String& url); |
|
||||||
|
|
||||||
void send(AsyncWebServerResponse *response); |
|
||||||
void send(int code, const String& contentType=String(), const String& content=String()); |
|
||||||
void send(FS &fs, const String& path, const String& contentType=String(), bool download=false, AwsTemplateProcessor callback=nullptr); |
|
||||||
void send(File content, const String& path, const String& contentType=String(), bool download=false, AwsTemplateProcessor callback=nullptr); |
|
||||||
void send(Stream &stream, const String& contentType, size_t len, AwsTemplateProcessor callback=nullptr); |
|
||||||
void send(const String& contentType, size_t len, AwsResponseFiller callback, AwsTemplateProcessor templateCallback=nullptr); |
|
||||||
void sendChunked(const String& contentType, AwsResponseFiller callback, AwsTemplateProcessor templateCallback=nullptr); |
|
||||||
void send_P(int code, const String& contentType, const uint8_t * content, size_t len, AwsTemplateProcessor callback=nullptr); |
|
||||||
void send_P(int code, const String& contentType, PGM_P content, AwsTemplateProcessor callback=nullptr); |
|
||||||
|
|
||||||
AsyncWebServerResponse *beginResponse(int code, const String& contentType=String(), const String& content=String()); |
|
||||||
AsyncWebServerResponse *beginResponse(FS &fs, const String& path, const String& contentType=String(), bool download=false, AwsTemplateProcessor callback=nullptr); |
|
||||||
AsyncWebServerResponse *beginResponse(File content, const String& path, const String& contentType=String(), bool download=false, AwsTemplateProcessor callback=nullptr); |
|
||||||
AsyncWebServerResponse *beginResponse(Stream &stream, const String& contentType, size_t len, AwsTemplateProcessor callback=nullptr); |
|
||||||
AsyncWebServerResponse *beginResponse(const String& contentType, size_t len, AwsResponseFiller callback, AwsTemplateProcessor templateCallback=nullptr); |
|
||||||
AsyncWebServerResponse *beginChunkedResponse(const String& contentType, AwsResponseFiller callback, AwsTemplateProcessor templateCallback=nullptr); |
|
||||||
AsyncResponseStream *beginResponseStream(const String& contentType, size_t bufferSize=1460); |
|
||||||
AsyncWebServerResponse *beginResponse_P(int code, const String& contentType, const uint8_t * content, size_t len, AwsTemplateProcessor callback=nullptr); |
|
||||||
AsyncWebServerResponse *beginResponse_P(int code, const String& contentType, PGM_P content, AwsTemplateProcessor callback=nullptr); |
|
||||||
|
|
||||||
size_t headers() const; // get header count
|
|
||||||
bool hasHeader(const String& name) const; // check if header exists
|
|
||||||
bool hasHeader(const __FlashStringHelper * data) const; // check if header exists
|
|
||||||
|
|
||||||
AsyncWebHeader* getHeader(const String& name) const; |
|
||||||
AsyncWebHeader* getHeader(const __FlashStringHelper * data) const; |
|
||||||
AsyncWebHeader* getHeader(size_t num) const; |
|
||||||
|
|
||||||
size_t params() const; // get arguments count
|
|
||||||
bool hasParam(const String& name, bool post=false, bool file=false) const; |
|
||||||
bool hasParam(const __FlashStringHelper * data, bool post=false, bool file=false) const; |
|
||||||
|
|
||||||
AsyncWebParameter* getParam(const String& name, bool post=false, bool file=false) const; |
|
||||||
AsyncWebParameter* getParam(const __FlashStringHelper * data, bool post, bool file) const;
|
|
||||||
AsyncWebParameter* getParam(size_t num) const; |
|
||||||
|
|
||||||
size_t args() const { return params(); } // get arguments count
|
|
||||||
const String& arg(const String& name) const; // get request argument value by name
|
|
||||||
const String& arg(const __FlashStringHelper * data) const; // get request argument value by F(name)
|
|
||||||
const String& arg(size_t i) const; // get request argument value by number
|
|
||||||
const String& argName(size_t i) const; // get request argument name by number
|
|
||||||
bool hasArg(const char* name) const; // check if argument exists
|
|
||||||
bool hasArg(const __FlashStringHelper * data) const; // check if F(argument) exists
|
|
||||||
|
|
||||||
const String& ASYNCWEBSERVER_REGEX_ATTRIBUTE pathArg(size_t i) const; |
|
||||||
|
|
||||||
const String& header(const char* name) const;// get request header value by name
|
|
||||||
const String& header(const __FlashStringHelper * data) const;// get request header value by F(name)
|
|
||||||
const String& header(size_t i) const; // get request header value by number
|
|
||||||
const String& headerName(size_t i) const; // get request header name by number
|
|
||||||
String urlDecode(const String& text) const; |
|
||||||
}; |
|
||||||
|
|
||||||
/*
|
|
||||||
* FILTER :: Callback to filter AsyncWebRewrite and AsyncWebHandler (done by the Server) |
|
||||||
* */ |
|
||||||
|
|
||||||
typedef std::function<bool(AsyncWebServerRequest *request)> ArRequestFilterFunction; |
|
||||||
|
|
||||||
bool ON_STA_FILTER(AsyncWebServerRequest *request); |
|
||||||
|
|
||||||
bool ON_AP_FILTER(AsyncWebServerRequest *request); |
|
||||||
|
|
||||||
/*
|
|
||||||
* REWRITE :: One instance can be handle any Request (done by the Server) |
|
||||||
* */ |
|
||||||
|
|
||||||
class AsyncWebRewrite { |
|
||||||
protected: |
|
||||||
String _from; |
|
||||||
String _toUrl; |
|
||||||
String _params; |
|
||||||
ArRequestFilterFunction _filter; |
|
||||||
public: |
|
||||||
AsyncWebRewrite(const char* from, const char* to): _from(from), _toUrl(to), _params(String()), _filter(NULL){ |
|
||||||
int index = _toUrl.indexOf('?'); |
|
||||||
if (index > 0) { |
|
||||||
_params = _toUrl.substring(index +1); |
|
||||||
_toUrl = _toUrl.substring(0, index); |
|
||||||
} |
|
||||||
} |
|
||||||
virtual ~AsyncWebRewrite(){} |
|
||||||
AsyncWebRewrite& setFilter(ArRequestFilterFunction fn) { _filter = fn; return *this; } |
|
||||||
bool filter(AsyncWebServerRequest *request) const { return _filter == NULL || _filter(request); } |
|
||||||
const String& from(void) const { return _from; } |
|
||||||
const String& toUrl(void) const { return _toUrl; } |
|
||||||
const String& params(void) const { return _params; } |
|
||||||
virtual bool match(AsyncWebServerRequest *request) { return from() == request->url() && filter(request); } |
|
||||||
}; |
|
||||||
|
|
||||||
/*
|
|
||||||
* HANDLER :: One instance can be attached to any Request (done by the Server) |
|
||||||
* */ |
|
||||||
|
|
||||||
class AsyncWebHandler { |
|
||||||
protected: |
|
||||||
ArRequestFilterFunction _filter; |
|
||||||
String _username; |
|
||||||
String _password; |
|
||||||
public: |
|
||||||
AsyncWebHandler():_username(""), _password(""){} |
|
||||||
AsyncWebHandler& setFilter(ArRequestFilterFunction fn) { _filter = fn; return *this; } |
|
||||||
AsyncWebHandler& setAuthentication(const char *username, const char *password){ _username = String(username);_password = String(password); return *this; }; |
|
||||||
bool filter(AsyncWebServerRequest *request){ return _filter == NULL || _filter(request); } |
|
||||||
virtual ~AsyncWebHandler(){} |
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request __attribute__((unused))){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request __attribute__((unused))){} |
|
||||||
virtual void handleUpload(AsyncWebServerRequest *request __attribute__((unused)), const String& filename __attribute__((unused)), size_t index __attribute__((unused)), uint8_t *data __attribute__((unused)), size_t len __attribute__((unused)), bool final __attribute__((unused))){} |
|
||||||
virtual void handleBody(AsyncWebServerRequest *request __attribute__((unused)), uint8_t *data __attribute__((unused)), size_t len __attribute__((unused)), size_t index __attribute__((unused)), size_t total __attribute__((unused))){} |
|
||||||
virtual bool isRequestHandlerTrivial(){return true;} |
|
||||||
}; |
|
||||||
|
|
||||||
/*
|
|
||||||
* RESPONSE :: One instance is created for each Request (attached by the Handler) |
|
||||||
* */ |
|
||||||
|
|
||||||
typedef enum { |
|
||||||
RESPONSE_SETUP, RESPONSE_HEADERS, RESPONSE_CONTENT, RESPONSE_WAIT_ACK, RESPONSE_END, RESPONSE_FAILED |
|
||||||
} WebResponseState; |
|
||||||
|
|
||||||
class AsyncWebServerResponse { |
|
||||||
protected: |
|
||||||
int _code; |
|
||||||
LinkedList<AsyncWebHeader *> _headers; |
|
||||||
String _contentType; |
|
||||||
size_t _contentLength; |
|
||||||
bool _sendContentLength; |
|
||||||
bool _chunked; |
|
||||||
size_t _headLength; |
|
||||||
size_t _sentLength; |
|
||||||
size_t _ackedLength; |
|
||||||
size_t _writtenLength; |
|
||||||
WebResponseState _state; |
|
||||||
const char* _responseCodeToString(int code); |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebServerResponse(); |
|
||||||
virtual ~AsyncWebServerResponse(); |
|
||||||
virtual void setCode(int code); |
|
||||||
virtual void setContentLength(size_t len); |
|
||||||
virtual void setContentType(const String& type); |
|
||||||
virtual void addHeader(const String& name, const String& value); |
|
||||||
virtual String _assembleHead(uint8_t version); |
|
||||||
virtual bool _started() const; |
|
||||||
virtual bool _finished() const; |
|
||||||
virtual bool _failed() const; |
|
||||||
virtual bool _sourceValid() const; |
|
||||||
virtual void _respond(AsyncWebServerRequest *request); |
|
||||||
virtual size_t _ack(AsyncWebServerRequest *request, size_t len, uint32_t time); |
|
||||||
}; |
|
||||||
|
|
||||||
/*
|
|
||||||
* SERVER :: One instance |
|
||||||
* */ |
|
||||||
|
|
||||||
typedef std::function<void(AsyncWebServerRequest *request)> ArRequestHandlerFunction; |
|
||||||
typedef std::function<void(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final)> ArUploadHandlerFunction; |
|
||||||
typedef std::function<void(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total)> ArBodyHandlerFunction; |
|
||||||
|
|
||||||
class AsyncWebServer { |
|
||||||
protected: |
|
||||||
AsyncServer _server; |
|
||||||
LinkedList<AsyncWebRewrite*> _rewrites; |
|
||||||
LinkedList<AsyncWebHandler*> _handlers; |
|
||||||
AsyncCallbackWebHandler* _catchAllHandler; |
|
||||||
|
|
||||||
public: |
|
||||||
AsyncWebServer(uint16_t port); |
|
||||||
~AsyncWebServer(); |
|
||||||
|
|
||||||
void begin(); |
|
||||||
void end(); |
|
||||||
|
|
||||||
#if ASYNC_TCP_SSL_ENABLED |
|
||||||
void onSslFileRequest(AcSSlFileHandler cb, void* arg); |
|
||||||
void beginSecure(const char *cert, const char *private_key_file, const char *password); |
|
||||||
#endif |
|
||||||
|
|
||||||
AsyncWebRewrite& addRewrite(AsyncWebRewrite* rewrite); |
|
||||||
bool removeRewrite(AsyncWebRewrite* rewrite); |
|
||||||
AsyncWebRewrite& rewrite(const char* from, const char* to); |
|
||||||
|
|
||||||
AsyncWebHandler& addHandler(AsyncWebHandler* handler); |
|
||||||
bool removeHandler(AsyncWebHandler* handler); |
|
||||||
|
|
||||||
AsyncCallbackWebHandler& on(const char* uri, ArRequestHandlerFunction onRequest); |
|
||||||
AsyncCallbackWebHandler& on(const char* uri, WebRequestMethodComposite method, ArRequestHandlerFunction onRequest); |
|
||||||
AsyncCallbackWebHandler& on(const char* uri, WebRequestMethodComposite method, ArRequestHandlerFunction onRequest, ArUploadHandlerFunction onUpload); |
|
||||||
AsyncCallbackWebHandler& on(const char* uri, WebRequestMethodComposite method, ArRequestHandlerFunction onRequest, ArUploadHandlerFunction onUpload, ArBodyHandlerFunction onBody); |
|
||||||
|
|
||||||
AsyncStaticWebHandler& serveStatic(const char* uri, fs::FS& fs, const char* path, const char* cache_control = NULL); |
|
||||||
|
|
||||||
void onNotFound(ArRequestHandlerFunction fn); //called when handler is not assigned
|
|
||||||
void onFileUpload(ArUploadHandlerFunction fn); //handle file uploads
|
|
||||||
void onRequestBody(ArBodyHandlerFunction fn); //handle posts with plain body content (JSON often transmitted this way as a request)
|
|
||||||
|
|
||||||
void reset(); //remove all writers and handlers, with onNotFound/onFileUpload/onRequestBody
|
|
||||||
|
|
||||||
void _handleDisconnect(AsyncWebServerRequest *request); |
|
||||||
void _attachHandler(AsyncWebServerRequest *request); |
|
||||||
void _rewriteRequest(AsyncWebServerRequest *request); |
|
||||||
}; |
|
||||||
|
|
||||||
class DefaultHeaders { |
|
||||||
using headers_t = LinkedList<AsyncWebHeader *>; |
|
||||||
headers_t _headers; |
|
||||||
|
|
||||||
DefaultHeaders() |
|
||||||
:_headers(headers_t([](AsyncWebHeader *h){ delete h; })) |
|
||||||
{} |
|
||||||
public: |
|
||||||
using ConstIterator = headers_t::ConstIterator; |
|
||||||
|
|
||||||
void addHeader(const String& name, const String& value){ |
|
||||||
_headers.add(new AsyncWebHeader(name, value)); |
|
||||||
}
|
|
||||||
|
|
||||||
ConstIterator begin() const { return _headers.begin(); } |
|
||||||
ConstIterator end() const { return _headers.end(); } |
|
||||||
|
|
||||||
DefaultHeaders(DefaultHeaders const &) = delete; |
|
||||||
DefaultHeaders &operator=(DefaultHeaders const &) = delete; |
|
||||||
static DefaultHeaders &Instance() { |
|
||||||
static DefaultHeaders instance; |
|
||||||
return instance; |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
#include "WebResponseImpl.h" |
|
||||||
#include "WebHandlerImpl.h" |
|
||||||
#include "AsyncWebSocket.h" |
|
||||||
#include "AsyncEventSource.h" |
|
||||||
|
|
||||||
#endif /* _AsyncWebServer_H_ */ |
|
@ -1,544 +0,0 @@ |
|||||||
#include "SPIFFSEditor.h" |
|
||||||
#include <FS.h> |
|
||||||
|
|
||||||
//File: edit.htm.gz, Size: 4151
|
|
||||||
#define edit_htm_gz_len 4151 |
|
||||||
const uint8_t edit_htm_gz[] PROGMEM = { |
|
||||||
0x1F, 0x8B, 0x08, 0x08, 0xB8, 0x94, 0xB1, 0x59, 0x00, 0x03, 0x65, 0x64, 0x69, 0x74, 0x2E, 0x68, |
|
||||||
0x74, 0x6D, 0x00, 0xB5, 0x3A, 0x0B, 0x7B, 0xDA, 0xB8, 0xB2, 0x7F, 0xC5, 0x71, 0xCF, 0x66, 0xED, |
|
||||||
0x83, 0x31, 0x90, 0xA4, 0xD9, 0xD6, 0xC4, 0xC9, 0x42, 0x92, 0x36, 0x6D, 0xF3, 0x6A, 0x80, 0xB6, |
|
||||||
0x69, 0x4F, 0xEE, 0x7E, 0xC2, 0x16, 0xA0, 0xC6, 0x96, 0x5D, 0x5B, 0x0E, 0x49, 0x59, 0xFE, 0xFB, |
|
||||||
0x9D, 0x91, 0x6C, 0xB0, 0x09, 0x69, 0x77, 0xCF, 0xBD, 0xBB, 0xDD, 0x2D, 0x92, 0x46, 0x33, 0x9A, |
|
||||||
0x19, 0xCD, 0x53, 0xDE, 0xBD, 0x8D, 0xA3, 0x8B, 0xC3, 0xFE, 0xF5, 0xE5, 0xB1, 0x36, 0x11, 0x61, |
|
||||||
0xB0, 0xBF, 0x87, 0x7F, 0x6B, 0x01, 0xE1, 0x63, 0x97, 0xF2, 0xFD, 0x3D, 0xC1, 0x44, 0x40, 0xF7, |
|
||||||
0x8F, 0x7B, 0x97, 0xDA, 0xB1, 0xCF, 0x44, 0x94, 0xEC, 0x35, 0xD4, 0xCA, 0x5E, 0x2A, 0x1E, 0x02, |
|
||||||
0xAA, 0x85, 0xD4, 0x67, 0xC4, 0x4D, 0xBD, 0x84, 0xC2, 0x66, 0xDB, 0x0B, 0x67, 0xDF, 0xEB, 0x8C, |
|
||||||
0xFB, 0xF4, 0xDE, 0xD9, 0x6E, 0x36, 0xDB, 0x71, 0x94, 0x32, 0xC1, 0x22, 0xEE, 0x90, 0x61, 0x1A, |
|
||||||
0x05, 0x99, 0xA0, 0xED, 0x80, 0x8E, 0x84, 0xF3, 0x3C, 0xBE, 0x6F, 0x0F, 0xA3, 0xC4, 0xA7, 0x89, |
|
||||||
0xD3, 0x8A, 0xEF, 0x35, 0x00, 0x31, 0x5F, 0x7B, 0xB6, 0xB3, 0xB3, 0xD3, 0x1E, 0x12, 0xEF, 0x76, |
|
||||||
0x9C, 0x44, 0x19, 0xF7, 0xEB, 0x5E, 0x14, 0x44, 0x89, 0xF3, 0x6C, 0xF4, 0x1C, 0xFF, 0xB4, 0x7D, |
|
||||||
0x96, 0xC6, 0x01, 0x79, 0x70, 0x78, 0xC4, 0x29, 0xE0, 0xDE, 0xD7, 0xD3, 0x09, 0xF1, 0xA3, 0xA9, |
|
||||||
0xD3, 0xD4, 0x9A, 0x5A, 0xAB, 0x09, 0x44, 0x92, 0xF1, 0x90, 0x18, 0x4D, 0x0B, 0xFF, 0xD8, 0x3B, |
|
||||||
0x66, 0x7B, 0x14, 0x71, 0x51, 0x4F, 0xD9, 0x77, 0xEA, 0xB4, 0xB6, 0xE0, 0x34, 0x39, 0x1D, 0x91, |
|
||||||
0x90, 0x05, 0x0F, 0x4E, 0x4A, 0x78, 0x5A, 0x4F, 0x69, 0xC2, 0x46, 0x6A, 0x79, 0x4A, 0xD9, 0x78, |
|
||||||
0x22, 0x9C, 0xDF, 0x9A, 0xCD, 0x39, 0xF0, 0xAF, 0x65, 0xC1, 0x2C, 0x60, 0x29, 0x20, 0xA3, 0x78, |
|
||||||
0xEA, 0x3C, 0x11, 0xC5, 0x4E, 0x53, 0xB1, 0xDE, 0x6C, 0x87, 0x24, 0x19, 0x33, 0x0E, 0x83, 0x98, |
|
||||||
0xF8, 0x3E, 0xE3, 0x63, 0x47, 0xA1, 0x05, 0x6C, 0xB6, 0x90, 0x36, 0xA1, 0x01, 0x11, 0xEC, 0x8E, |
|
||||||
0xB6, 0x43, 0xC6, 0xEB, 0x53, 0xE6, 0x8B, 0x89, 0xB3, 0x0B, 0x3C, 0xB6, 0xBD, 0x2C, 0x49, 0x41, |
|
||||||
0xA6, 0x38, 0x62, 0x5C, 0xD0, 0x44, 0xA2, 0xA5, 0x31, 0xE1, 0xB3, 0x5C, 0x54, 0x54, 0x40, 0x21, |
|
||||||
0x27, 0xE3, 0x01, 0xE3, 0xB4, 0x3E, 0x0C, 0x22, 0xEF, 0x76, 0x71, 0xD2, 0x6E, 0x7C, 0x9F, 0x9F, |
|
||||||
0xE5, 0x4C, 0xA2, 0x3B, 0x9A, 0xCC, 0x96, 0xEA, 0x92, 0xD8, 0x15, 0x60, 0x85, 0x34, 0xA5, 0x74, |
|
||||||
0x6E, 0x8B, 0xBB, 0x0C, 0xA0, 0x96, 0xFC, 0x05, 0x29, 0x17, 0xFC, 0x2F, 0x45, 0x5A, 0x11, 0x5C, |
|
||||||
0xA1, 0x30, 0x1E, 0x67, 0x62, 0xF6, 0xF8, 0x2A, 0xA3, 0x98, 0x78, 0x4C, 0x3C, 0xA0, 0xFC, 0xB0, |
|
||||||
0x6D, 0x86, 0xBA, 0x04, 0xAC, 0x24, 0x24, 0x81, 0x86, 0x3A, 0xD7, 0x3E, 0xD0, 0xC4, 0x27, 0x9C, |
|
||||||
0x58, 0x9D, 0x84, 0x91, 0xC0, 0xEA, 0x2D, 0xB5, 0x5E, 0x0F, 0xA3, 0xEF, 0xF5, 0x0C, 0xC6, 0x30, |
|
||||||
0x0F, 0xA8, 0x27, 0x94, 0x92, 0xE1, 0x1E, 0x86, 0xB7, 0x4C, 0x3C, 0x06, 0x3C, 0x5A, 0x28, 0xA9, |
|
||||||
0x4B, 0x2A, 0x69, 0xA2, 0x2E, 0xB0, 0x25, 0xD5, 0x83, 0x1C, 0x4B, 0xC9, 0x95, 0x50, 0xF5, 0x61, |
|
||||||
0x24, 0x44, 0x14, 0x4A, 0x93, 0x5B, 0x08, 0xAC, 0x49, 0xAB, 0x79, 0xF1, 0xE8, 0x46, 0xD6, 0x6B, |
|
||||||
0xBF, 0x44, 0xBE, 0x0D, 0x7A, 0x15, 0xCC, 0x23, 0x41, 0x9D, 0x04, 0x6C, 0xCC, 0x9D, 0x90, 0xF9, |
|
||||||
0x7E, 0x40, 0x4B, 0x56, 0xEB, 0x64, 0x49, 0x60, 0xF8, 0x44, 0x10, 0x87, 0x85, 0x64, 0x4C, 0x1B, |
|
||||||
0x31, 0x1F, 0x03, 0x34, 0xA5, 0xBB, 0x3B, 0x16, 0xFB, 0xD0, 0xBD, 0xB8, 0x9A, 0x36, 0xDF, 0xBD, |
|
||||||
0x1E, 0x47, 0x1D, 0xF8, 0xE7, 0xBC, 0x37, 0x98, 0x1C, 0x0F, 0xC6, 0x30, 0xEA, 0xE2, 0xB4, 0xF3, |
|
||||||
0xFE, 0xB0, 0xF3, 0x1E, 0x7E, 0x0E, 0x5B, 0xB5, 0xAF, 0xA3, 0x6F, 0xB8, 0xD0, 0x7D, 0xED, 0x77, |
|
||||||
0xFB, 0x83, 0xE3, 0x4E, 0xE7, 0x5D, 0xE3, 0xCD, 0xF9, 0xF4, 0xE3, 0xBB, 0x5D, 0x04, 0x77, 0x83, |
|
||||||
0xE6, 0xD5, 0x87, 0x49, 0x73, 0xB0, 0xF5, 0x32, 0xF4, 0x4F, 0xFC, 0x89, 0x17, 0x0E, 0x3A, 0xEF, |
|
||||||
0x3F, 0x5E, 0xDD, 0x5D, 0x87, 0x83, 0x71, 0xEF, 0x63, 0x6B, 0xF2, 0x79, 0xEB, 0x43, 0xEF, 0xF3, |
|
||||||
0xC7, 0x57, 0xB7, 0xF4, 0xD3, 0xC9, 0xDB, 0xCF, 0xFD, 0x29, 0x20, 0x1C, 0x45, 0xBD, 0xC1, 0x55, |
|
||||||
0xF7, 0x43, 0x77, 0xFC, 0xB9, 0xEB, 0x1D, 0xDF, 0x0F, 0x83, 0xF3, 0xEE, 0xEB, 0xCE, 0xB0, 0xB3, |
|
||||||
0xE5, 0x51, 0x3A, 0xEE, 0x5F, 0x75, 0xB3, 0x37, 0xEF, 0x2E, 0xC6, 0x8C, 0x4D, 0x7A, 0x9F, 0xCF, |
|
||||||
0xFB, 0xDE, 0xE1, 0xF3, 0xD3, 0xC1, 0x49, 0x87, 0x4D, 0xCE, 0xDF, 0x5E, 0x35, 0x6F, 0x5F, 0xBF, |
|
||||||
0x3B, 0x3C, 0xF2, 0xAE, 0xDF, 0x5E, 0xEF, 0x1E, 0x6D, 0x37, 0x7E, 0xFB, 0xED, 0xCC, 0xBF, 0x60, |
|
||||||
0xBC, 0x7F, 0xF7, 0xBD, 0x33, 0x3E, 0x9C, 0xBE, 0x78, 0x48, 0xFB, 0x93, 0x37, 0x77, 0xBC, 0xF1, |
|
||||||
0x21, 0xFA, 0xFA, 0xE6, 0xE1, 0x0C, 0xFE, 0xBB, 0xBC, 0xAC, 0x0D, 0x7B, 0xAD, 0x74, 0xF0, 0xFE, |
|
||||||
0xCD, 0x87, 0xAD, 0xF4, 0xE5, 0xF3, 0xB8, 0x7B, 0x74, 0x74, 0x17, 0x0E, 0x2F, 0x1B, 0xA1, 0x7F, |
|
||||||
0x3B, 0x12, 0x2F, 0xB6, 0x45, 0x7C, 0x3D, 0xCE, 0x3E, 0x7F, 0x7B, 0xFE, 0x76, 0xD2, 0xB8, 0xA0, |
|
||||||
0xE4, 0x7A, 0x52, 0x7B, 0xF8, 0xFE, 0xF0, 0x62, 0xD2, 0x3F, 0xB9, 0x3B, 0x0F, 0xC8, 0xFD, 0xF9, |
|
||||||
0xB9, 0xF7, 0x3D, 0xAC, 0x05, 0xE4, 0xE5, 0x45, 0x3F, 0x20, 0x49, 0x6B, 0xE0, 0x77, 0x1A, 0xB5, |
|
||||||
0xC3, 0xAD, 0xCE, 0x8E, 0x48, 0xAE, 0x0E, 0xF9, 0xD1, 0xF6, 0xD7, 0xDE, 0x8B, 0x6E, 0xB7, 0x15, |
|
||||||
0x0D, 0xBF, 0x6D, 0xBD, 0xBE, 0xDD, 0x7D, 0x3D, 0xD8, 0x7D, 0x3F, 0x7C, 0xDF, 0xE9, 0xED, 0x74, |
|
||||||
0x07, 0xE4, 0xBA, 0xF7, 0xBE, 0x33, 0xDA, 0x19, 0x4E, 0x26, 0xEF, 0xDE, 0xF5, 0x5F, 0xF9, 0x9D, |
|
||||||
0xEF, 0x49, 0xE7, 0x62, 0xDA, 0xB9, 0x3F, 0x1E, 0x74, 0x4E, 0x6A, 0xEF, 0x8E, 0xCF, 0x9A, 0xAD, |
|
||||||
0xDE, 0xF5, 0xF6, 0xF8, 0x6C, 0x77, 0xDA, 0x4D, 0x8F, 0x3B, 0xEF, 0xBB, 0xCD, 0xF1, 0xDB, 0x5A, |
|
||||||
0x48, 0x3E, 0x47, 0x87, 0xDB, 0xE3, 0x37, 0xBB, 0xEC, 0xF2, 0x9A, 0x74, 0xDE, 0x74, 0xDF, 0xA6, |
|
||||||
0xEC, 0x2A, 0x3C, 0x19, 0x34, 0x3B, 0x9D, 0xD3, 0x0B, 0xFA, 0xEA, 0x70, 0x9B, 0xBC, 0xDB, 0xF2, |
|
||||||
0x3E, 0x82, 0xFE, 0x07, 0x9F, 0xE8, 0x6F, 0xB5, 0xCE, 0xF4, 0xA2, 0x19, 0x78, 0x2F, 0x69, 0xFF, |
|
||||||
0xE4, 0xBA, 0x2F, 0x6F, 0xE7, 0x38, 0x78, 0xD5, 0xBF, 0xED, 0x65, 0xEF, 0xC3, 0xC3, 0x43, 0x53, |
|
||||||
0xE3, 0x51, 0x3D, 0xA1, 0x31, 0x25, 0xA2, 0x1C, 0xAE, 0x16, 0xFE, 0x01, 0xB6, 0xB5, 0xB4, 0xC2, |
|
||||||
0xDC, 0x4F, 0x05, 0xBD, 0x17, 0x75, 0x9F, 0x7A, 0x51, 0x42, 0xE4, 0x1E, 0x40, 0xA0, 0x09, 0x9A, |
|
||||||
0xD8, 0xFC, 0x77, 0x19, 0x3F, 0x35, 0x15, 0x3F, 0x35, 0xC2, 0x7D, 0xCD, 0x28, 0x1C, 0x01, 0x83, |
|
||||||
0x87, 0x4F, 0xEF, 0x98, 0x47, 0xEB, 0x31, 0xBB, 0xA7, 0x41, 0x5D, 0x22, 0x3B, 0x4D, 0x73, 0x26, |
|
||||||
0xFD, 0xAD, 0xD8, 0x46, 0x38, 0x98, 0x9A, 0xA4, 0x5A, 0x2C, 0xF8, 0x5F, 0x89, 0x47, 0x21, 0xB0, |
|
||||||
0x81, 0xCB, 0x84, 0xF8, 0xAB, 0x7C, 0x27, 0x4A, 0xEA, 0xC3, 0x6C, 0x3C, 0x62, 0xF7, 0xE0, 0xD0, |
|
||||||
0x23, 0xC6, 0x99, 0xA0, 0x5A, 0x2B, 0x9D, 0xFF, 0x5E, 0x90, 0xB9, 0xA5, 0x0F, 0xA3, 0x84, 0x84, |
|
||||||
0x34, 0xD5, 0xFE, 0x22, 0x99, 0xD9, 0x28, 0x89, 0xC2, 0x65, 0x10, 0x99, 0x8B, 0xA8, 0x34, 0x99, |
|
||||||
0xCF, 0x9F, 0x65, 0x71, 0x10, 0x11, 0x10, 0x73, 0x4D, 0xE4, 0x50, 0xF1, 0x34, 0x91, 0x6E, 0xB5, |
|
||||||
0x88, 0xAB, 0xB9, 0x9B, 0x6D, 0xA1, 0x5B, 0x96, 0xDD, 0x7A, 0x6B, 0x67, 0xE9, 0xBA, 0x75, 0xB9, |
|
||||||
0x17, 0xE3, 0xFD, 0x9A, 0x4C, 0x81, 0xF1, 0xA0, 0x14, 0xEE, 0x9E, 0x09, 0x50, 0xE9, 0x13, 0x87, |
|
||||||
0xCB, 0x43, 0xF2, 0xC8, 0xB0, 0x60, 0x40, 0x05, 0xEA, 0x96, 0x8C, 0xD4, 0x85, 0x24, 0xB0, 0x6F, |
|
||||||
0xFE, 0x8C, 0xCA, 0xBC, 0x67, 0x3D, 0x8B, 0x13, 0xB8, 0x0D, 0x3A, 0xFD, 0x11, 0xCD, 0x42, 0xA6, |
|
||||||
0x2A, 0x6D, 0x45, 0x53, 0x65, 0xBC, 0x5C, 0x84, 0x65, 0xDA, 0x93, 0xBC, 0x16, 0xA4, 0x1F, 0x4B, |
|
||||||
0x05, 0xE0, 0x05, 0x37, 0xCF, 0x91, 0x9B, 0x1F, 0x6A, 0x75, 0x7B, 0xF7, 0x97, 0x9C, 0x87, 0x9D, |
|
||||||
0xE6, 0x2F, 0x73, 0x3B, 0xDF, 0x5B, 0xA4, 0xE4, 0x56, 0x13, 0xFE, 0x29, 0x32, 0xEF, 0x8B, 0x25, |
|
||||||
0x0B, 0xC3, 0xE7, 0xF8, 0xA7, 0x60, 0x10, 0xE9, 0x94, 0x80, 0xDB, 0x3B, 0x2F, 0x5F, 0xF8, 0xC3, |
|
||||||
0x02, 0x98, 0x0B, 0xF6, 0x24, 0x3C, 0x21, 0x3E, 0xCB, 0x52, 0xE7, 0x79, 0xF3, 0x97, 0x5C, 0x9F, |
|
||||||
0x5B, 0x3B, 0x28, 0xFB, 0xE2, 0x2E, 0x71, 0xB2, 0xB4, 0xD8, 0x34, 0x66, 0x5C, 0xDB, 0x4A, 0x35, |
|
||||||
0xBC, 0x6F, 0x92, 0x2C, 0x0C, 0xB3, 0x92, 0xED, 0xE7, 0xBF, 0x2F, 0x4D, 0x13, 0xF7, 0xCF, 0x9A, |
|
||||||
0xBF, 0xCC, 0x44, 0x02, 0xD9, 0x64, 0x04, 0xB9, 0xC6, 0x49, 0x22, 0x41, 0x04, 0x35, 0x9A, 0xE6, |
|
||||||
0x1C, 0x84, 0x5B, 0x03, 0xD8, 0xDE, 0x6D, 0xFA, 0x74, 0x6C, 0xCE, 0xE7, 0x7B, 0x0D, 0x99, 0xD7, |
|
||||||
0xA0, 0x6C, 0xF1, 0x12, 0x16, 0x8B, 0xFD, 0x51, 0xC6, 0x3D, 0xE4, 0x41, 0x1B, 0x53, 0x83, 0x9A, |
|
||||||
0xB3, 0x84, 0x8A, 0x2C, 0xE1, 0x9A, 0x1F, 0x79, 0x19, 0x1A, 0xBB, 0x3D, 0xA6, 0xE2, 0x58, 0xD9, |
|
||||||
0x7D, 0xF7, 0xE1, 0x8D, 0x0F, 0x3B, 0xE6, 0x0B, 0x04, 0x6F, 0x2D, 0x02, 0x38, 0x30, 0x9C, 0x97, |
|
||||||
0xE3, 0x54, 0xF6, 0x43, 0x82, 0x01, 0x22, 0xEF, 0xE8, 0x83, 0x41, 0x2D, 0xB1, 0x40, 0xA4, 0x36, |
|
||||||
0xAE, 0x1B, 0xC5, 0x2E, 0x80, 0x71, 0x73, 0x76, 0x07, 0x4A, 0x20, 0x2E, 0xFD, 0x22, 0x6E, 0x2C, |
|
||||||
0xE6, 0x72, 0xF8, 0x69, 0xE7, 0xBB, 0xC9, 0x1E, 0x3B, 0xA8, 0xB7, 0x1C, 0xB2, 0xCF, 0x0E, 0x5A, |
|
||||||
0xE0, 0x5E, 0x65, 0x6E, 0xE4, 0xB9, 0xAF, 0x58, 0x40, 0x07, 0xB9, 0xC3, 0xE1, 0x31, 0x48, 0x6C, |
|
||||||
0xB1, 0x85, 0x28, 0xE2, 0x5B, 0xCD, 0xE6, 0x86, 0x4B, 0x0F, 0x48, 0x00, 0x39, 0xCC, 0xD0, 0x8F, |
|
||||||
0xAF, 0xAE, 0x2E, 0xAE, 0xBE, 0xE8, 0x35, 0x5A, 0xD3, 0x6F, 0x1C, 0x4D, 0xAF, 0x71, 0xD3, 0x11, |
|
||||||
0x76, 0x42, 0x47, 0x09, 0x4D, 0x27, 0x97, 0x44, 0x4C, 0x8C, 0xD4, 0xBE, 0x23, 0x41, 0x56, 0x16, |
|
||||||
0x84, 0xA1, 0xDC, 0xC8, 0xA2, 0x70, 0x39, 0x9D, 0x6A, 0xAF, 0x40, 0xCD, 0x47, 0x90, 0xEA, 0xDA, |
|
||||||
0xC2, 0x26, 0x71, 0x4C, 0xB9, 0x6F, 0xE8, 0x31, 0x20, 0xEA, 0x16, 0x35, 0xAD, 0x84, 0x7E, 0xCB, |
|
||||||
0x68, 0x2A, 0x52, 0x1B, 0x2C, 0xD7, 0xD0, 0x2F, 0x07, 0x7D, 0xDD, 0xD2, 0x1B, 0xE8, 0x47, 0x3A, |
|
||||||
0xF0, 0x46, 0xCC, 0x39, 0x52, 0x89, 0x5C, 0xD0, 0xA4, 0x3E, 0xCC, 0xC0, 0xA0, 0xB8, 0x6E, 0xB6, |
|
||||||
0x23, 0x9B, 0x71, 0x4E, 0x93, 0x93, 0xFE, 0xD9, 0xA9, 0xAB, 0x5F, 0x29, 0x46, 0xB4, 0x53, 0x28, |
|
||||||
0x48, 0x74, 0x4B, 0x5E, 0x51, 0x7E, 0xC8, 0xE1, 0x84, 0x05, 0xBE, 0x11, 0x99, 0x6D, 0x24, 0xE1, |
|
||||||
0x49, 0x12, 0xB2, 0x40, 0x01, 0x0A, 0x9E, 0x2D, 0x1E, 0x62, 0xEA, 0xEA, 0x23, 0x50, 0x86, 0x6E, |
|
||||||
0x79, 0x76, 0x98, 0x05, 0x82, 0xC5, 0x01, 0x75, 0x37, 0x5A, 0x30, 0xE3, 0x60, 0x41, 0xAE, 0x8E, |
|
||||||
0xB9, 0x19, 0x61, 0xCC, 0x77, 0x75, 0x15, 0xA1, 0xF2, 0xB8, 0xB6, 0xEE, 0x14, 0x4F, 0x9D, 0x92, |
|
||||||
0x56, 0x4E, 0x49, 0xCB, 0xB8, 0x4A, 0xE0, 0x34, 0x3F, 0x18, 0xC3, 0x3C, 0xCE, 0xD4, 0x51, 0x05, |
|
||||||
0xCC, 0xA7, 0x23, 0x02, 0x9C, 0x7C, 0x40, 0x6D, 0xBA, 0x7A, 0x63, 0xDD, 0x41, 0xA9, 0x3A, 0xC8, |
|
||||||
0xAF, 0x6A, 0xC4, 0x2F, 0x6B, 0x44, 0xDD, 0xEE, 0x3A, 0x64, 0x5F, 0x21, 0x07, 0x55, 0xE4, 0xA0, |
|
||||||
0x8C, 0x7C, 0x28, 0x8D, 0x64, 0x1D, 0x72, 0xA0, 0x90, 0x93, 0x8A, 0x88, 0x89, 0x14, 0x51, 0x85, |
|
||||||
0xBD, 0x3A, 0x6A, 0x13, 0x05, 0xD2, 0xAD, 0xA4, 0x22, 0x66, 0x62, 0x83, 0x97, 0x92, 0x61, 0x40, |
|
||||||
0x7D, 0x77, 0xA3, 0x09, 0x33, 0x2C, 0xB6, 0xDD, 0xAD, 0xE6, 0x9A, 0x33, 0x12, 0x75, 0x46, 0x56, |
|
||||||
0x65, 0x30, 0x2B, 0x33, 0xA8, 0xF5, 0xC8, 0x1D, 0xD5, 0xD6, 0x31, 0x98, 0x99, 0x56, 0x60, 0x47, |
|
||||||
0xDC, 0x0B, 0x98, 0x77, 0xEB, 0x2E, 0xBD, 0xC5, 0x9C, 0xB1, 0x85, 0x85, 0x5A, 0x5C, 0x06, 0xBA, |
|
||||||
0x01, 0x94, 0x5E, 0x8B, 0xA5, 0x7C, 0x80, 0xFA, 0x9E, 0x5B, 0xD9, 0x5A, 0x02, 0xDC, 0xA6, 0xF7, |
|
||||||
0xD4, 0x3B, 0x8C, 0xC2, 0x90, 0xA0, 0xED, 0xA6, 0xC0, 0x41, 0x3E, 0xD1, 0xCD, 0xB9, 0x15, 0xAD, |
|
||||||
0xC5, 0x79, 0xC2, 0x45, 0x2C, 0x7F, 0x3D, 0x8B, 0x23, 0x03, 0x5C, 0xCE, 0xF5, 0x6C, 0xD4, 0x61, |
|
||||||
0x6A, 0x83, 0x1E, 0xC7, 0x62, 0xF2, 0x13, 0x17, 0x2A, 0x0C, 0x54, 0xA2, 0x7C, 0x69, 0xDE, 0x58, |
|
||||||
0x0B, 0x91, 0x56, 0x7C, 0xEA, 0xA2, 0xB7, 0xE2, 0x54, 0xA8, 0xBC, 0x8A, 0x5D, 0x9A, 0x4B, 0x1D, |
|
||||||
0x94, 0x61, 0xB9, 0xBD, 0x2F, 0xA0, 0xFA, 0x7C, 0x0E, 0xE7, 0x01, 0xFF, 0x13, 0x68, 0xF9, 0xE8, |
|
||||||
0x5F, 0x17, 0x60, 0xC9, 0xA3, 0x34, 0x78, 0x8B, 0xBB, 0x0D, 0xE3, 0xC0, 0xF9, 0x8F, 0x6D, 0x7C, |
|
||||||
0xF9, 0x1F, 0xFB, 0xA6, 0x66, 0x9A, 0x07, 0xFF, 0x6A, 0x48, 0x0D, 0x1B, 0xC2, 0xFC, 0xD2, 0xBA, |
|
||||||
0xB1, 0x08, 0x80, 0xED, 0x7F, 0x9B, 0xFF, 0xB1, 0x25, 0xB8, 0x02, 0x6B, 0xDF, 0x45, 0x90, 0x49, |
|
||||||
0xF0, 0x24, 0x34, 0xB0, 0x68, 0xA4, 0x91, 0xCD, 0x4D, 0x43, 0xB8, 0xA4, 0x72, 0x8D, 0x35, 0x51, |
|
||||||
0xD3, 0x6D, 0x88, 0x53, 0x50, 0x5B, 0xAC, 0x04, 0xBF, 0x3E, 0x24, 0x7A, 0x15, 0x5B, 0x17, 0x00, |
|
||||||
0xC9, 0x3D, 0xCA, 0x0C, 0x3D, 0x22, 0x97, 0x52, 0xCB, 0x0C, 0x02, 0x42, 0xA7, 0x89, 0xE7, 0x2A, |
|
||||||
0xAD, 0x1D, 0x14, 0x30, 0x17, 0xA2, 0xE0, 0xBC, 0x1C, 0x2D, 0x15, 0xEA, 0xAA, 0xFD, 0x17, 0x0A, |
|
||||||
0xA3, 0xD6, 0x12, 0x8A, 0x04, 0x31, 0xAD, 0xD8, 0x79, 0xC6, 0x72, 0x75, 0x4C, 0x59, 0xBA, 0x35, |
|
||||||
0x59, 0x5D, 0x96, 0xAD, 0x04, 0xAE, 0x2F, 0x8D, 0xFE, 0xD7, 0x3D, 0x16, 0x8E, 0xB5, 0x12, 0x3F, |
|
||||||
0xF8, 0x97, 0xFB, 0x2B, 0x46, 0xE4, 0xCD, 0x3F, 0xBC, 0x21, 0x70, 0x05, 0xA6, 0x41, 0x6D, 0x1E, |
|
||||||
0x4D, 0x0D, 0xB3, 0xF6, 0xAB, 0xAE, 0x49, 0x8A, 0xAE, 0x1E, 0x92, 0xFB, 0xBC, 0xA7, 0xC4, 0x8C, |
|
||||||
0xD7, 0xD6, 0x70, 0x5E, 0xB4, 0x28, 0xF9, 0x82, 0xEC, 0xE6, 0x48, 0x26, 0xA2, 0xB6, 0x56, 0x64, |
|
||||||
0x52, 0xD5, 0xCA, 0xE8, 0x5A, 0x63, 0xFF, 0xD7, 0x4A, 0x40, 0xB7, 0x98, 0xBA, 0x4E, 0x15, 0x8C, |
|
||||||
0xB3, 0x00, 0x1C, 0x93, 0x3E, 0x1D, 0x69, 0x03, 0x26, 0x03, 0x75, 0x35, 0x46, 0x5A, 0x81, 0xC1, |
|
||||||
0xCC, 0x03, 0xC3, 0x2B, 0xFB, 0xF3, 0x1E, 0x16, 0xBF, 0xFB, 0x97, 0xAA, 0xAA, 0x81, 0xD4, 0x8B, |
|
||||||
0x33, 0x5D, 0x59, 0x59, 0xD5, 0x4B, 0xE0, 0xD2, 0x08, 0xA0, 0x5B, 0x8B, 0x3C, 0x3A, 0x8C, 0xFC, |
|
||||||
0x87, 0x52, 0xF6, 0x4D, 0xBB, 0x0F, 0x87, 0x01, 0x49, 0xD3, 0x73, 0xB8, 0x01, 0x43, 0xF7, 0x42, |
|
||||||
0x50, 0xB8, 0xB2, 0xC2, 0xFD, 0xE6, 0xE6, 0x66, 0x15, 0x29, 0xA1, 0x21, 0x14, 0xDB, 0x8A, 0x2B, |
|
||||||
0xF0, 0x49, 0xD3, 0xF1, 0x81, 0x30, 0x18, 0xD2, 0x1A, 0xC6, 0xF0, 0x25, 0xE3, 0x47, 0x5C, 0x71, |
|
||||||
0xF4, 0xF4, 0x22, 0xA6, 0xFC, 0x33, 0xDC, 0x95, 0x32, 0xCB, 0x1A, 0xAD, 0xA6, 0x68, 0xFA, 0x8F, |
|
||||||
0xD8, 0x3E, 0xCA, 0x0D, 0x76, 0xC1, 0x7A, 0xBA, 0x56, 0xA1, 0xFC, 0x9F, 0x61, 0xB9, 0x94, 0x28, |
|
||||||
0xD6, 0x70, 0x9C, 0x40, 0x80, 0x5A, 0xC3, 0x31, 0xC4, 0x1A, 0x41, 0x17, 0xFC, 0x26, 0x6B, 0xF9, |
|
||||||
0xCD, 0xFE, 0x19, 0x7E, 0x97, 0x76, 0x1E, 0x15, 0x25, 0x91, 0xAA, 0xAF, 0x50, 0x02, 0x9F, 0xDD, |
|
||||||
0xE9, 0xA6, 0x15, 0xB9, 0x55, 0x0A, 0x50, 0x1B, 0x46, 0x41, 0xD0, 0x8F, 0xE2, 0x83, 0x27, 0xD6, |
|
||||||
0x9D, 0xC5, 0x7A, 0x31, 0xC8, 0xD9, 0x5C, 0x6E, 0xB1, 0xBC, 0xB5, 0x44, 0x4F, 0xA1, 0xEC, 0x5F, |
|
||||||
0x4B, 0x15, 0x01, 0x3F, 0x23, 0x8B, 0x7B, 0xAC, 0xD4, 0xA5, 0x36, 0x28, 0x0F, 0x56, 0x3F, 0xD5, |
|
||||||
0x3C, 0xCB, 0x5F, 0xCC, 0xAE, 0x6B, 0x51, 0x9B, 0xC0, 0x38, 0x57, 0x92, 0x8B, 0x4A, 0xB2, 0xC8, |
|
||||||
0x13, 0x01, 0xA8, 0x58, 0xC7, 0x2E, 0xC4, 0x4D, 0x6B, 0x7A, 0x7C, 0xBF, 0x5C, 0x83, 0xC2, 0xDF, |
|
||||||
0xF5, 0xD5, 0x12, 0x33, 0x08, 0xC4, 0xD3, 0x95, 0x4B, 0x29, 0x5F, 0x37, 0x29, 0x8A, 0x0E, 0x62, |
|
||||||
0x47, 0xA3, 0x51, 0x4A, 0xC5, 0x47, 0x0C, 0x49, 0x56, 0xB2, 0x98, 0x9F, 0xC8, 0x90, 0x04, 0x8C, |
|
||||||
0x45, 0x3C, 0x8C, 0xB2, 0x94, 0x46, 0x99, 0xA8, 0xA4, 0x16, 0x63, 0x21, 0xCC, 0x5E, 0xFA, 0xE7, |
|
||||||
0x9F, 0x8B, 0xC9, 0x7E, 0x5A, 0x0B, 0x96, 0xD3, 0xEB, 0x3D, 0xBF, 0x34, 0xD9, 0xF7, 0x6B, 0x89, |
|
||||||
0xB9, 0x7A, 0xE9, 0xFF, 0x67, 0x4B, 0x21, 0x65, 0x4B, 0xF1, 0xB0, 0x54, 0x2E, 0x62, 0x62, 0x29, |
|
||||||
0xE6, 0xC9, 0x82, 0x91, 0x97, 0x7C, 0x16, 0x0D, 0x1A, 0x2B, 0x25, 0x55, 0x9E, 0x97, 0x7D, 0x95, |
|
||||||
0x43, 0x40, 0x59, 0x71, 0xE5, 0x35, 0x11, 0x06, 0x34, 0xE0, 0x63, 0x64, 0xF2, 0x41, 0xEB, 0xA7, |
|
||||||
0xD1, 0x94, 0x26, 0x87, 0x24, 0xA5, 0x06, 0x24, 0xCD, 0x65, 0xDC, 0x41, 0xA8, 0xE9, 0x04, 0xEB, |
|
||||||
0x76, 0x6D, 0x6E, 0x12, 0x05, 0xCE, 0x33, 0x77, 0xC4, 0xB1, 0x26, 0x03, 0xF9, 0xB2, 0xCA, 0x09, |
|
||||||
0xD4, 0xC6, 0xBE, 0x12, 0xA4, 0x3E, 0x52, 0x25, 0xA8, 0x61, 0x5A, 0xD0, 0x76, 0xC0, 0x35, 0x5F, |
|
||||||
0x26, 0x51, 0x4C, 0xC6, 0xB2, 0x07, 0x83, 0x35, 0x74, 0x0F, 0xA4, 0x66, 0x6D, 0x34, 0x91, 0x60, |
|
||||||
0xA9, 0x73, 0x29, 0xFC, 0x66, 0xD9, 0xC2, 0x70, 0x4B, 0x57, 0xC9, 0xB0, 0xBD, 0xF4, 0xA5, 0x35, |
|
||||||
0x59, 0x83, 0xE0, 0x0B, 0x6C, 0x62, 0xE0, 0x1E, 0x68, 0x64, 0xF2, 0x7B, 0x00, 0x77, 0x6B, 0xB6, |
|
||||||
0xA3, 0x3D, 0xD6, 0x8E, 0x6A, 0x35, 0x53, 0x55, 0xE9, 0xAE, 0x0B, 0x6D, 0x4E, 0x74, 0x23, 0x0B, |
|
||||||
0x4B, 0x10, 0xAA, 0x9A, 0x59, 0x0C, 0x38, 0x1B, 0x81, 0xAA, 0xBA, 0xC0, 0x11, 0xD6, 0x98, 0x66, |
|
||||||
0xA9, 0x23, 0xF1, 0x97, 0x1D, 0xC9, 0x13, 0xB5, 0x07, 0x95, 0xF5, 0x05, 0xD4, 0x31, 0xAB, 0x25, |
|
||||||
0x86, 0x30, 0xD3, 0x29, 0x13, 0xDE, 0x04, 0x03, 0x90, 0x07, 0x5A, 0xD5, 0x05, 0x14, 0xB5, 0x8E, |
|
||||||
0x1C, 0x4D, 0x44, 0xB8, 0x1C, 0x05, 0xF9, 0xF0, 0x6B, 0x9A, 0x0F, 0xBC, 0xB4, 0x18, 0xDD, 0x97, |
|
||||||
0x80, 0x50, 0xD2, 0xE6, 0xE0, 0x88, 0x8F, 0xF2, 0x21, 0xF4, 0xB2, 0x05, 0x9D, 0x02, 0x58, 0xFC, |
|
||||||
0xC6, 0x71, 0x3E, 0x8A, 0x27, 0xC5, 0x68, 0x42, 0xEF, 0x17, 0x78, 0x51, 0x01, 0xF5, 0xA9, 0xEE, |
|
||||||
0x28, 0x1B, 0xDB, 0x68, 0xCE, 0xF3, 0x41, 0x6B, 0x29, 0x7F, 0xF0, 0xFF, 0x28, 0x7F, 0xCC, 0xC7, |
|
||||||
0x85, 0x34, 0x71, 0x31, 0x1A, 0xB3, 0x42, 0x96, 0x61, 0x18, 0xFF, 0x90, 0x93, 0xA4, 0xD4, 0x13, |
|
||||||
0x97, 0x7A, 0x5A, 0xF1, 0xB3, 0xB6, 0x53, 0x98, 0x8E, 0x31, 0xAA, 0xF8, 0xE3, 0xC8, 0xF6, 0xF0, |
|
||||||
0xF7, 0x3C, 0xF2, 0x65, 0x6D, 0x69, 0x5A, 0xA1, 0x31, 0x82, 0x3A, 0x57, 0x37, 0xCB, 0x7E, 0x9A, |
|
||||||
0xFD, 0xB7, 0xAD, 0xE8, 0xD1, 0xF1, 0xE9, 0x71, 0xFF, 0xB8, 0x5C, 0x38, 0x23, 0xE7, 0x25, 0x93, |
|
||||||
0x8A, 0x2B, 0x5D, 0xFA, 0xB2, 0x22, 0x80, 0x02, 0x1B, 0x45, 0x01, 0x7B, 0xDD, 0xDC, 0x54, 0x7E, |
|
||||||
0xF1, 0xB6, 0x77, 0x71, 0x6E, 0xC7, 0x24, 0x01, 0x8F, 0x24, 0x15, 0xE6, 0xC2, 0x82, 0x44, 0xF9, |
|
||||||
0xE0, 0xD7, 0xC7, 0xA5, 0x72, 0x5D, 0x7E, 0x61, 0x70, 0xC4, 0xDC, 0x52, 0xA7, 0xA9, 0x7E, 0x78, |
|
||||||
0xE2, 0x62, 0x5D, 0x99, 0xBF, 0x04, 0x41, 0x72, 0x1A, 0x2D, 0x13, 0x55, 0x11, 0x67, 0x46, 0xE5, |
|
||||||
0x30, 0x2F, 0xEE, 0xB2, 0x75, 0x0D, 0xD3, 0xC8, 0xB4, 0xC4, 0x84, 0xA5, 0xE5, 0x46, 0xA5, 0x12, |
|
||||||
0x14, 0xFE, 0xA2, 0xB6, 0xE7, 0x8B, 0x91, 0x24, 0xB7, 0x5A, 0x73, 0xAB, 0x6F, 0x41, 0x2A, 0x3E, |
|
||||||
0x58, 0x04, 0x23, 0x66, 0x39, 0xDB, 0x16, 0x77, 0xA3, 0x43, 0xEE, 0x61, 0x5C, 0x7F, 0xBA, 0x35, |
|
||||||
0x78, 0xD2, 0x3C, 0x79, 0x61, 0x9E, 0xFC, 0xB1, 0x7B, 0x2E, 0x1C, 0x45, 0xF9, 0xDA, 0xE2, 0x98, |
|
||||||
0xF6, 0x10, 0x58, 0xBB, 0x6D, 0x2F, 0x7D, 0x18, 0x20, 0xD2, 0x83, 0xCB, 0x00, 0xF4, 0x63, 0x58, |
|
||||||
0xFF, 0x4A, 0xEE, 0x88, 0x7A, 0x09, 0xAA, 0xA2, 0xAD, 0x73, 0x54, 0xD8, 0xEE, 0xFD, 0x81, 0xA3, |
|
||||||
0xF2, 0xCE, 0x65, 0x18, 0x48, 0x97, 0xC3, 0x92, 0x37, 0x8B, 0x75, 0xC1, 0x61, 0x19, 0x31, 0x64, |
|
||||||
0x6C, 0x00, 0xE3, 0xCD, 0x5D, 0x49, 0x13, 0xD5, 0x1C, 0xB4, 0xF0, 0x1B, 0x08, 0x8A, 0x4F, 0x39, |
|
||||||
0xCE, 0x9A, 0x38, 0xAD, 0x62, 0x72, 0xC5, 0x23, 0xC8, 0x4A, 0x67, 0x89, 0xC0, 0x6E, 0x10, 0x0D, |
|
||||||
0x0D, 0x7C, 0x64, 0x9A, 0xA1, 0xB6, 0x1D, 0x3E, 0x37, 0xD7, 0xBC, 0xD9, 0x54, 0xFA, 0x4B, 0x62, |
|
||||||
0x79, 0xD5, 0xB0, 0x8B, 0x1C, 0x56, 0xCC, 0x75, 0x7D, 0x1F, 0xF4, 0xA3, 0x4E, 0x29, 0xAF, 0x48, |
|
||||||
0xA4, 0x53, 0xD1, 0x83, 0xC4, 0x86, 0xA2, 0x41, 0xBE, 0x91, 0x40, 0x44, 0x72, 0x4A, 0x33, 0x5D, |
|
||||||
0xC7, 0xCA, 0xD2, 0x0B, 0x28, 0x49, 0x7A, 0xB2, 0x73, 0x95, 0x49, 0x6B, 0x25, 0x06, 0xFE, 0xC8, |
|
||||||
0xD7, 0xF0, 0xC7, 0xA1, 0xD0, 0xA3, 0x83, 0x9B, 0x49, 0x2B, 0x83, 0xA4, 0x23, 0x64, 0x83, 0xA9, |
|
||||||
0x37, 0xE4, 0xBB, 0xA8, 0x2D, 0x2F, 0xCB, 0xB4, 0x16, 0x50, 0x70, 0x71, 0x83, 0xBB, 0x11, 0x30, |
|
||||||
0x52, 0x5A, 0xC4, 0x9E, 0x94, 0xA8, 0xC7, 0x8F, 0x10, 0x1F, 0x53, 0x4A, 0x20, 0x06, 0x20, 0xA6, |
|
||||||
0x40, 0xD0, 0xA7, 0x42, 0x8A, 0x54, 0xE6, 0x92, 0x53, 0x2A, 0x20, 0xCA, 0x48, 0xCD, 0xE2, 0xC1, |
|
||||||
0x85, 0x78, 0xD4, 0x46, 0xD6, 0x80, 0xFD, 0xDC, 0xBD, 0x73, 0x33, 0xDE, 0x90, 0x68, 0x09, 0x56, |
|
||||||
0x36, 0x3D, 0x9A, 0xA6, 0x52, 0x5C, 0x54, 0xC7, 0x19, 0xF8, 0xA8, 0xA1, 0x03, 0x5A, 0x23, 0x84, |
|
||||||
0x11, 0x1E, 0x84, 0x8A, 0x01, 0x40, 0x7F, 0x42, 0xC3, 0x1C, 0x22, 0x70, 0x08, 0x20, 0x82, 0xA0, |
|
||||||
0x7F, 0x49, 0x0D, 0xF7, 0x64, 0x05, 0xC9, 0xF8, 0xD8, 0x6D, 0x35, 0xF0, 0x9D, 0x66, 0x95, 0xEC, |
|
||||||
0x20, 0xA5, 0xBD, 0x68, 0x24, 0xFA, 0x64, 0x98, 0x1A, 0x50, 0x00, 0xAC, 0xD9, 0x01, 0xA0, 0x1E, |
|
||||||
0x24, 0x5E, 0x63, 0x2B, 0x3F, 0xEF, 0x04, 0x2A, 0xBB, 0x00, 0xAB, 0xBB, 0x8E, 0x87, 0x5F, 0x39, |
|
||||||
0x4F, 0x19, 0xA7, 0x39, 0x26, 0x00, 0x7B, 0x93, 0x68, 0x7A, 0x99, 0x30, 0x2E, 0xCE, 0x64, 0x1B, |
|
||||||
0x6A, 0x6C, 0xB4, 0xE4, 0xF5, 0xA9, 0x87, 0x15, 0x79, 0x3F, 0xC5, 0x8B, 0xCB, 0x0C, 0xF3, 0xBA, |
|
||||||
0x53, 0x79, 0x77, 0xB1, 0x86, 0x70, 0x21, 0x50, 0x66, 0x38, 0xB3, 0x29, 0x74, 0xB0, 0xFA, 0xA1, |
|
||||||
0x48, 0x82, 0x7A, 0x4F, 0xB7, 0x42, 0xE2, 0xC1, 0x44, 0xED, 0x81, 0xF9, 0xDC, 0xC2, 0xD8, 0xE1, |
|
||||||
0x94, 0x83, 0x5A, 0x0A, 0xB5, 0x02, 0x45, 0xC6, 0x95, 0xCD, 0x98, 0x35, 0x1D, 0x6A, 0x58, 0x88, |
|
||||||
0x61, 0xE0, 0xAF, 0xFE, 0x05, 0x0F, 0x1E, 0x1C, 0xC8, 0x55, 0x3F, 0xE1, 0x23, 0xE3, 0x7E, 0xF4, |
|
||||||
0x23, 0x3E, 0x3E, 0xAF, 0xF0, 0xF1, 0x79, 0x1D, 0x1F, 0xB4, 0xAA, 0x3C, 0x98, 0x0C, 0x80, 0xEC, |
|
||||||
0x19, 0xE1, 0x64, 0x4C, 0x13, 0x58, 0xC0, 0x43, 0x50, 0x25, 0x7F, 0x8B, 0xB3, 0x84, 0xFE, 0x98, |
|
||||||
0xB3, 0xDE, 0x84, 0x8D, 0xC4, 0x23, 0xFE, 0x8A, 0xD5, 0xFF, 0x82, 0x4B, 0x3C, 0x70, 0x3D, 0x97, |
|
||||||
0x79, 0x6D, 0x5A, 0x49, 0x28, 0x3F, 0x7E, 0x2B, 0x91, 0x7E, 0xE4, 0x42, 0x78, 0xA9, 0x38, 0xC8, |
|
||||||
0xDF, 0xB7, 0xF4, 0x00, 0xBC, 0x11, 0xF8, 0x29, 0x35, 0x75, 0xBC, 0x0B, 0xA5, 0xFC, 0x29, 0x30, |
|
||||||
0x64, 0xA8, 0xC0, 0x47, 0xDD, 0xD9, 0xDC, 0x12, 0xAE, 0x01, 0x8A, 0xF1, 0xA3, 0x29, 0xB0, 0xEA, |
|
||||||
0xC9, 0x02, 0xD7, 0x9E, 0x40, 0x26, 0x04, 0x91, 0xE0, 0x48, 0xC8, 0xA7, 0x8D, 0x2F, 0x07, 0x9B, |
|
||||||
0x37, 0x35, 0xC8, 0x43, 0x2E, 0xFC, 0x98, 0x2E, 0x0C, 0x36, 0x6F, 0xFE, 0x6D, 0x36, 0xC6, 0xCC, |
|
||||||
0x5A, 0x76, 0xA4, 0x96, 0x4C, 0xF6, 0xF4, 0x0B, 0xBF, 0x71, 0x09, 0x48, 0x5D, 0x49, 0x78, 0x45, |
|
||||||
0x34, 0x03, 0x6B, 0x43, 0x61, 0xE1, 0x07, 0xFF, 0x47, 0x09, 0xF8, 0x91, 0x9E, 0x07, 0xCE, 0xBD, |
|
||||||
0xE6, 0x3D, 0x5E, 0x2F, 0x3E, 0x85, 0xE9, 0x56, 0xE9, 0xC1, 0x4A, 0xC7, 0xEF, 0x53, 0x3A, 0x76, |
|
||||||
0x59, 0xA2, 0x14, 0x4A, 0x14, 0x59, 0x88, 0x1A, 0x6A, 0x50, 0x0E, 0x51, 0x98, 0x89, 0x17, 0xCD, |
|
||||||
0x81, 0x02, 0x9B, 0x73, 0x34, 0x5B, 0x3A, 0x02, 0x0F, 0xF4, 0xF5, 0x45, 0xEE, 0xFC, 0x74, 0x76, |
|
||||||
0x7A, 0x22, 0x44, 0x7C, 0xA5, 0x62, 0x22, 0xD0, 0xAA, 0x2E, 0x2C, 0x2F, 0xCF, 0x9C, 0x89, 0xE4, |
|
||||||
0xA1, 0x28, 0x75, 0x30, 0x31, 0x28, 0x87, 0xFE, 0x74, 0x31, 0xFC, 0x0A, 0x71, 0xD6, 0xD0, 0xCF, |
|
||||||
0x52, 0x48, 0x58, 0x5B, 0x36, 0xA2, 0xF7, 0xFB, 0x97, 0xF6, 0xAE, 0xDD, 0x84, 0xBA, 0x00, 0xB4, |
|
||||||
0x0A, 0x69, 0x19, 0xEE, 0x7D, 0xFE, 0xB7, 0x90, 0xB7, 0xFF, 0x1E, 0x32, 0x83, 0xA8, 0x95, 0x42, |
|
||||||
0x58, 0x2A, 0xF0, 0xAB, 0xB8, 0x93, 0x24, 0x9A, 0x4A, 0xB4, 0xE3, 0x24, 0xC1, 0x4B, 0xE9, 0x43, |
|
||||||
0x85, 0xA2, 0x0D, 0x61, 0x31, 0xA5, 0x89, 0xE6, 0x47, 0x34, 0xD5, 0x78, 0x24, 0xB4, 0x34, 0x8B, |
|
||||||
0x63, 0x68, 0x5C, 0x56, 0xF4, 0x61, 0xEB, 0xC5, 0xEB, 0xCB, 0xFB, 0x8C, 0x66, 0xD4, 0xCF, 0x97, |
|
||||||
0x69, 0x52, 0xD1, 0x0B, 0x56, 0x50, 0xDF, 0x10, 0xEE, 0x7E, 0xB9, 0xC9, 0xEB, 0xA9, 0x8C, 0x73, |
|
||||||
0x8C, 0xA2, 0x1B, 0x2D, 0x35, 0x07, 0xE9, 0x26, 0x40, 0xD5, 0xE5, 0x59, 0x10, 0xCC, 0xDB, 0x2B, |
|
||||||
0xB4, 0xA0, 0xF1, 0x8A, 0x44, 0x24, 0x9F, 0xCB, 0x67, 0x7F, 0xE4, 0xC9, 0xA9, 0xE2, 0x82, 0x50, |
|
||||||
0xF2, 0x54, 0xA9, 0x36, 0xAD, 0x0D, 0x63, 0x83, 0x6A, 0x8C, 0xA7, 0x82, 0x70, 0x0F, 0xAF, 0x51, |
|
||||||
0xE9, 0xC2, 0x2C, 0x6A, 0x29, 0xDC, 0xDE, 0x46, 0x5F, 0xCB, 0x6D, 0xE9, 0x89, 0x7C, 0x2A, 0x25, |
|
||||||
0xE3, 0xAE, 0xAE, 0x63, 0x55, 0x45, 0xB1, 0x3E, 0x25, 0x61, 0x5A, 0x26, 0x5B, 0x54, 0x06, 0x26, |
|
||||||
0x77, 0x0B, 0x70, 0x9B, 0x06, 0x29, 0x1C, 0xBD, 0x7E, 0x7F, 0xCE, 0x46, 0xD1, 0xCE, 0x11, 0x80, |
|
||||||
0x69, 0xC5, 0x3E, 0x93, 0xD7, 0xE0, 0x24, 0xCC, 0x73, 0x07, 0x32, 0xE9, 0x4A, 0x03, 0x0E, 0xA9, |
|
||||||
0x98, 0x44, 0xFE, 0x81, 0x7E, 0xA0, 0x3B, 0x3A, 0xFC, 0xBB, 0x09, 0x35, 0x47, 0xCD, 0xA5, 0xD0, |
|
||||||
0xA4, 0xFA, 0x74, 0x70, 0xF5, 0x06, 0xC2, 0x53, 0x0C, 0xA5, 0x01, 0x17, 0x50, 0x34, 0xD7, 0x74, |
|
||||||
0x7C, 0x7A, 0x7D, 0x0C, 0x29, 0xC8, 0x7F, 0x21, 0x37, 0x66, 0xBB, 0xAA, 0x6C, 0xB8, 0xF3, 0xEA, |
|
||||||
0x75, 0x56, 0x2E, 0x03, 0x7A, 0x61, 0x8C, 0x58, 0x0F, 0x29, 0x7E, 0xFB, 0x7B, 0xF4, 0x9E, 0x8D, |
|
||||||
0x15, 0xD2, 0x6A, 0x5D, 0x6F, 0xCE, 0x76, 0x90, 0x67, 0x89, 0xD5, 0x43, 0x2C, 0x70, 0x97, 0x1F, |
|
||||||
0x29, 0x59, 0x95, 0x35, 0xDC, 0xF6, 0x48, 0x10, 0xE0, 0xC7, 0x5A, 0x03, 0x1B, 0x6A, 0x22, 0xB2, |
|
||||||
0xD4, 0x42, 0x22, 0x29, 0x08, 0x90, 0xD2, 0x3E, 0x84, 0x39, 0xD3, 0x92, 0x65, 0x86, 0xB2, 0xA1, |
|
||||||
0xBC, 0xFF, 0xC5, 0x9A, 0xA3, 0x64, 0x46, 0xE8, 0xCE, 0xF9, 0x6C, 0x73, 0x53, 0xD8, 0x85, 0x99, |
|
||||||
0x18, 0x05, 0x52, 0x8A, 0x01, 0x1C, 0x9A, 0x7D, 0x68, 0x2D, 0x8C, 0xB2, 0x90, 0x58, 0xAB, 0x3D, |
|
||||||
0xD2, 0xB6, 0x51, 0x55, 0x03, 0x54, 0x7C, 0x46, 0x01, 0x03, 0xCE, 0xB2, 0x24, 0x80, 0xA8, 0x8B, |
|
||||||
0x39, 0xBA, 0xB2, 0x2D, 0xC5, 0xBA, 0xD0, 0x84, 0x0E, 0xEC, 0x67, 0xC8, 0x12, 0x95, 0x97, 0xAD, |
|
||||||
0xA2, 0x27, 0x12, 0xC5, 0x77, 0x95, 0x9E, 0xC8, 0x6F, 0xE5, 0x84, 0xAA, 0xC8, 0x77, 0x88, 0x2F, |
|
||||||
0x13, 0x5C, 0xD4, 0xD1, 0x13, 0xA0, 0x24, 0x83, 0x52, 0x34, 0x60, 0x2A, 0x2C, 0x37, 0xEE, 0xEB, |
|
||||||
0xD3, 0xE9, 0xB4, 0x8E, 0xDF, 0x6A, 0xEB, 0x70, 0x82, 0xB2, 0x02, 0x5F, 0x5F, 0xC7, 0x21, 0x47, |
|
||||||
0x15, 0x58, 0xF8, 0x6E, 0xE1, 0xAC, 0xBA, 0xE8, 0x42, 0x7F, 0x2B, 0xDE, 0xD4, 0xAA, 0xD2, 0x59, |
|
||||||
0xE1, 0x73, 0x79, 0xDB, 0x7B, 0x3B, 0x2B, 0x20, 0x32, 0xC4, 0xAF, 0xB2, 0x90, 0x69, 0x20, 0x0D, |
|
||||||
0x3B, 0xE5, 0x46, 0x56, 0x25, 0x85, 0x65, 0x5C, 0xB0, 0xE3, 0x2C, 0x9D, 0x18, 0x33, 0x60, 0xDD, |
|
||||||
0x11, 0x96, 0xD2, 0x95, 0x43, 0x2D, 0x65, 0xB7, 0x0E, 0xB7, 0x0A, 0xFB, 0x70, 0x30, 0x83, 0x94, |
|
||||||
0x79, 0xFB, 0xF3, 0x4F, 0x39, 0x5B, 0xDE, 0xF6, 0x92, 0x62, 0x71, 0xE1, 0xF3, 0xFC, 0xA9, 0x35, |
|
||||||
0xAF, 0x69, 0xA5, 0xD1, 0xAF, 0xC4, 0x97, 0xBD, 0x46, 0xFE, 0x19, 0x3B, 0xFF, 0x9C, 0xAD, 0x81, |
|
||||||
0xB1, 0x43, 0x23, 0x2A, 0xDC, 0x4C, 0x8C, 0xEA, 0x2F, 0x34, 0xE6, 0x63, 0x79, 0x29, 0xBF, 0x2D, |
|
||||||
0xA0, 0x54, 0xA9, 0xD3, 0x68, 0x78, 0x3E, 0xFF, 0x9A, 0x42, 0x19, 0x1D, 0x65, 0xFE, 0x28, 0x20, |
|
||||||
0x09, 0xC5, 0x82, 0xA3, 0x41, 0xBE, 0x92, 0xFB, 0x46, 0xC0, 0x86, 0x69, 0x03, 0x93, 0x6D, 0xCB, |
|
||||||
0xDE, 0xB2, 0x77, 0x71, 0x64, 0x7F, 0x4D, 0xF7, 0x57, 0x4F, 0xD8, 0x5F, 0x34, 0x69, 0x58, 0x0B, |
|
||||||
0xE7, 0xB5, 0xAB, 0x8A, 0x4D, 0x6A, 0x83, 0xFB, 0xC4, 0xA7, 0x70, 0x3D, 0x6F, 0xB3, 0xCC, 0xB6, |
|
||||||
0x1A, 0xE4, 0x5F, 0x60, 0xD4, 0x31, 0xBA, 0x95, 0x2F, 0x92, 0xF4, 0x81, 0x7B, 0x18, 0x5B, 0x17, |
|
||||||
0x54, 0x26, 0x70, 0x49, 0xD5, 0x87, 0x34, 0xB9, 0xD3, 0x9C, 0x2F, 0x39, 0xC3, 0xB7, 0x3C, 0xA8, |
|
||||||
0x03, 0xE4, 0x37, 0x9C, 0x72, 0x39, 0xB0, 0xBF, 0x07, 0x5D, 0x33, 0x2A, 0x41, 0x79, 0xB1, 0x26, |
|
||||||
0x9B, 0xE6, 0x7C, 0x02, 0x82, 0x01, 0x70, 0xB1, 0xA3, 0x48, 0xCD, 0x2B, 0xCB, 0x98, 0x9B, 0x57, |
|
||||||
0x96, 0x54, 0xE2, 0x5F, 0x59, 0xCC, 0xDB, 0x9F, 0xFC, 0xDB, 0x4C, 0xF9, 0x7F, 0x5B, 0x28, 0x36, |
|
||||||
0x32, 0xF9, 0xE1, 0x09, 0xF7, 0x56, 0x3F, 0x45, 0xAD, 0x47, 0x51, 0xBB, 0xF7, 0xFF, 0x17, 0x53, |
|
||||||
0xE8, 0x9D, 0x36, 0x92, 0x29, 0x00, 0x00 |
|
||||||
}; |
|
||||||
|
|
||||||
#define SPIFFS_MAXLENGTH_FILEPATH 32 |
|
||||||
const char *excludeListFile = "/.exclude.files"; |
|
||||||
|
|
||||||
typedef struct ExcludeListS { |
|
||||||
char *item; |
|
||||||
ExcludeListS *next; |
|
||||||
} ExcludeList; |
|
||||||
|
|
||||||
static ExcludeList *excludes = NULL; |
|
||||||
|
|
||||||
static bool matchWild(const char *pattern, const char *testee) { |
|
||||||
const char *nxPat = NULL, *nxTst = NULL; |
|
||||||
|
|
||||||
while (*testee) { |
|
||||||
if (( *pattern == '?' ) || (*pattern == *testee)){ |
|
||||||
pattern++;testee++; |
|
||||||
continue; |
|
||||||
} |
|
||||||
if (*pattern=='*'){ |
|
||||||
nxPat=pattern++; nxTst=testee; |
|
||||||
continue; |
|
||||||
} |
|
||||||
if (nxPat){
|
|
||||||
pattern = nxPat+1; testee=++nxTst; |
|
||||||
continue; |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
while (*pattern=='*'){pattern++;}
|
|
||||||
return (*pattern == 0); |
|
||||||
} |
|
||||||
|
|
||||||
static bool addExclude(const char *item){ |
|
||||||
size_t len = strlen(item); |
|
||||||
if(!len){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
ExcludeList *e = (ExcludeList *)malloc(sizeof(ExcludeList)); |
|
||||||
if(!e){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
e->item = (char *)malloc(len+1); |
|
||||||
if(!e->item){ |
|
||||||
free(e); |
|
||||||
return false; |
|
||||||
} |
|
||||||
memcpy(e->item, item, len+1); |
|
||||||
e->next = excludes; |
|
||||||
excludes = e; |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
static void loadExcludeList(fs::FS &_fs, const char *filename){ |
|
||||||
static char linebuf[SPIFFS_MAXLENGTH_FILEPATH]; |
|
||||||
fs::File excludeFile=_fs.open(filename, "r"); |
|
||||||
if(!excludeFile){ |
|
||||||
//addExclude("/*.js.gz");
|
|
||||||
return; |
|
||||||
} |
|
||||||
#ifdef ESP32 |
|
||||||
if(excludeFile.isDirectory()){ |
|
||||||
excludeFile.close(); |
|
||||||
return; |
|
||||||
} |
|
||||||
#endif |
|
||||||
if (excludeFile.size() > 0){ |
|
||||||
uint8_t idx; |
|
||||||
bool isOverflowed = false; |
|
||||||
while (excludeFile.available()){ |
|
||||||
linebuf[0] = '\0'; |
|
||||||
idx = 0; |
|
||||||
int lastChar; |
|
||||||
do { |
|
||||||
lastChar = excludeFile.read(); |
|
||||||
if(lastChar != '\r'){ |
|
||||||
linebuf[idx++] = (char) lastChar; |
|
||||||
} |
|
||||||
} while ((lastChar >= 0) && (lastChar != '\n') && (idx < SPIFFS_MAXLENGTH_FILEPATH)); |
|
||||||
|
|
||||||
if(isOverflowed){ |
|
||||||
isOverflowed = (lastChar != '\n'); |
|
||||||
continue; |
|
||||||
} |
|
||||||
isOverflowed = (idx >= SPIFFS_MAXLENGTH_FILEPATH); |
|
||||||
linebuf[idx-1] = '\0'; |
|
||||||
if(!addExclude(linebuf)){ |
|
||||||
excludeFile.close(); |
|
||||||
return; |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
excludeFile.close(); |
|
||||||
} |
|
||||||
|
|
||||||
static bool isExcluded(fs::FS &_fs, const char *filename) { |
|
||||||
if(excludes == NULL){ |
|
||||||
loadExcludeList(_fs, excludeListFile); |
|
||||||
} |
|
||||||
ExcludeList *e = excludes; |
|
||||||
while(e){ |
|
||||||
if (matchWild(e->item, filename)){ |
|
||||||
return true; |
|
||||||
} |
|
||||||
e = e->next; |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
// WEB HANDLER IMPLEMENTATION
|
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
SPIFFSEditor::SPIFFSEditor(const fs::FS& fs, const String& username, const String& password) |
|
||||||
#else |
|
||||||
SPIFFSEditor::SPIFFSEditor(const String& username, const String& password, const fs::FS& fs) |
|
||||||
#endif |
|
||||||
:_fs(fs) |
|
||||||
,_username(username) |
|
||||||
,_password(password) |
|
||||||
,_authenticated(false) |
|
||||||
,_startTime(0) |
|
||||||
{} |
|
||||||
|
|
||||||
bool SPIFFSEditor::canHandle(AsyncWebServerRequest *request){ |
|
||||||
if(request->url().equalsIgnoreCase("/edit")){ |
|
||||||
if(request->method() == HTTP_GET){ |
|
||||||
if(request->hasParam("list")) |
|
||||||
return true; |
|
||||||
if(request->hasParam("edit")){ |
|
||||||
request->_tempFile = _fs.open(request->arg("edit"), "r"); |
|
||||||
if(!request->_tempFile){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
#ifdef ESP32 |
|
||||||
if(request->_tempFile.isDirectory()){ |
|
||||||
request->_tempFile.close(); |
|
||||||
return false; |
|
||||||
} |
|
||||||
#endif |
|
||||||
} |
|
||||||
if(request->hasParam("download")){ |
|
||||||
request->_tempFile = _fs.open(request->arg("download"), "r"); |
|
||||||
if(!request->_tempFile){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
#ifdef ESP32 |
|
||||||
if(request->_tempFile.isDirectory()){ |
|
||||||
request->_tempFile.close(); |
|
||||||
return false; |
|
||||||
} |
|
||||||
#endif |
|
||||||
} |
|
||||||
request->addInterestingHeader("If-Modified-Since"); |
|
||||||
return true; |
|
||||||
} |
|
||||||
else if(request->method() == HTTP_POST) |
|
||||||
return true; |
|
||||||
else if(request->method() == HTTP_DELETE) |
|
||||||
return true; |
|
||||||
else if(request->method() == HTTP_PUT) |
|
||||||
return true; |
|
||||||
|
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
void SPIFFSEditor::handleRequest(AsyncWebServerRequest *request){ |
|
||||||
if(_username.length() && _password.length() && !request->authenticate(_username.c_str(), _password.c_str())) |
|
||||||
return request->requestAuthentication(); |
|
||||||
|
|
||||||
if(request->method() == HTTP_GET){ |
|
||||||
if(request->hasParam("list")){ |
|
||||||
String path = request->getParam("list")->value(); |
|
||||||
#ifdef ESP32 |
|
||||||
File dir = _fs.open(path); |
|
||||||
#else |
|
||||||
Dir dir = _fs.openDir(path); |
|
||||||
#endif |
|
||||||
path = String(); |
|
||||||
String output = "["; |
|
||||||
#ifdef ESP32 |
|
||||||
File entry = dir.openNextFile(); |
|
||||||
while(entry){ |
|
||||||
#else |
|
||||||
while(dir.next()){ |
|
||||||
fs::File entry = dir.openFile("r"); |
|
||||||
#endif |
|
||||||
if (isExcluded(_fs, entry.name())) { |
|
||||||
#ifdef ESP32 |
|
||||||
entry = dir.openNextFile(); |
|
||||||
#endif |
|
||||||
continue; |
|
||||||
} |
|
||||||
if (output != "[") output += ','; |
|
||||||
output += "{\"type\":\""; |
|
||||||
output += "file"; |
|
||||||
output += "\",\"name\":\""; |
|
||||||
output += String(entry.name()); |
|
||||||
output += "\",\"size\":"; |
|
||||||
output += String(entry.size()); |
|
||||||
output += "}"; |
|
||||||
#ifdef ESP32 |
|
||||||
entry = dir.openNextFile(); |
|
||||||
#else |
|
||||||
entry.close(); |
|
||||||
#endif |
|
||||||
} |
|
||||||
#ifdef ESP32 |
|
||||||
dir.close(); |
|
||||||
#endif |
|
||||||
output += "]"; |
|
||||||
request->send(200, "application/json", output); |
|
||||||
output = String(); |
|
||||||
} |
|
||||||
else if(request->hasParam("edit") || request->hasParam("download")){ |
|
||||||
request->send(request->_tempFile, request->_tempFile.name(), String(), request->hasParam("download")); |
|
||||||
} |
|
||||||
else { |
|
||||||
const char * buildTime = __DATE__ " " __TIME__ " GMT"; |
|
||||||
if (request->header("If-Modified-Since").equals(buildTime)) { |
|
||||||
request->send(304); |
|
||||||
} else { |
|
||||||
AsyncWebServerResponse *response = request->beginResponse_P(200, "text/html", edit_htm_gz, edit_htm_gz_len); |
|
||||||
response->addHeader("Content-Encoding", "gzip"); |
|
||||||
response->addHeader("Last-Modified", buildTime); |
|
||||||
request->send(response); |
|
||||||
} |
|
||||||
} |
|
||||||
} else if(request->method() == HTTP_DELETE){ |
|
||||||
if(request->hasParam("path", true)){ |
|
||||||
_fs.remove(request->getParam("path", true)->value()); |
|
||||||
request->send(200, "", "DELETE: "+request->getParam("path", true)->value()); |
|
||||||
} else |
|
||||||
request->send(404); |
|
||||||
} else if(request->method() == HTTP_POST){ |
|
||||||
if(request->hasParam("data", true, true) && _fs.exists(request->getParam("data", true, true)->value())) |
|
||||||
request->send(200, "", "UPLOADED: "+request->getParam("data", true, true)->value()); |
|
||||||
else |
|
||||||
request->send(500); |
|
||||||
} else if(request->method() == HTTP_PUT){ |
|
||||||
if(request->hasParam("path", true)){ |
|
||||||
String filename = request->getParam("path", true)->value(); |
|
||||||
if(_fs.exists(filename)){ |
|
||||||
request->send(200); |
|
||||||
} else { |
|
||||||
fs::File f = _fs.open(filename, "w"); |
|
||||||
if(f){ |
|
||||||
f.write((uint8_t)0x00); |
|
||||||
f.close(); |
|
||||||
request->send(200, "", "CREATE: "+filename); |
|
||||||
} else { |
|
||||||
request->send(500); |
|
||||||
} |
|
||||||
} |
|
||||||
} else |
|
||||||
request->send(400); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
void SPIFFSEditor::handleUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final){ |
|
||||||
if(!index){ |
|
||||||
if(!_username.length() || request->authenticate(_username.c_str(),_password.c_str())){ |
|
||||||
_authenticated = true; |
|
||||||
request->_tempFile = _fs.open(filename, "w"); |
|
||||||
_startTime = millis(); |
|
||||||
} |
|
||||||
} |
|
||||||
if(_authenticated && request->_tempFile){ |
|
||||||
if(len){ |
|
||||||
request->_tempFile.write(data,len); |
|
||||||
} |
|
||||||
if(final){ |
|
||||||
request->_tempFile.close(); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
@ -1,24 +0,0 @@ |
|||||||
#ifndef SPIFFSEditor_H_ |
|
||||||
#define SPIFFSEditor_H_ |
|
||||||
#include <ESPAsyncWebServer.h> |
|
||||||
|
|
||||||
class SPIFFSEditor: public AsyncWebHandler { |
|
||||||
private: |
|
||||||
fs::FS _fs; |
|
||||||
String _username; |
|
||||||
String _password;
|
|
||||||
bool _authenticated; |
|
||||||
uint32_t _startTime; |
|
||||||
public: |
|
||||||
#ifdef ESP32 |
|
||||||
SPIFFSEditor(const fs::FS& fs, const String& username=String(), const String& password=String()); |
|
||||||
#else |
|
||||||
SPIFFSEditor(const String& username=String(), const String& password=String(), const fs::FS& fs=SPIFFS); |
|
||||||
#endif |
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request) override final; |
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request) override final; |
|
||||||
virtual void handleUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) override final; |
|
||||||
virtual bool isRequestHandlerTrivial() override final {return false;} |
|
||||||
}; |
|
||||||
|
|
||||||
#endif |
|
@ -1,193 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#ifndef STRINGARRAY_H_ |
|
||||||
#define STRINGARRAY_H_ |
|
||||||
|
|
||||||
#include "stddef.h" |
|
||||||
#include "WString.h" |
|
||||||
|
|
||||||
template <typename T> |
|
||||||
class LinkedListNode { |
|
||||||
T _value; |
|
||||||
public: |
|
||||||
LinkedListNode<T>* next; |
|
||||||
LinkedListNode(const T val): _value(val), next(nullptr) {} |
|
||||||
~LinkedListNode(){} |
|
||||||
const T& value() const { return _value; }; |
|
||||||
T& value(){ return _value; } |
|
||||||
}; |
|
||||||
|
|
||||||
template <typename T, template<typename> class Item = LinkedListNode> |
|
||||||
class LinkedList { |
|
||||||
public: |
|
||||||
typedef Item<T> ItemType; |
|
||||||
typedef std::function<void(const T&)> OnRemove; |
|
||||||
typedef std::function<bool(const T&)> Predicate; |
|
||||||
private: |
|
||||||
ItemType* _root; |
|
||||||
OnRemove _onRemove; |
|
||||||
|
|
||||||
class Iterator { |
|
||||||
ItemType* _node; |
|
||||||
public: |
|
||||||
Iterator(ItemType* current = nullptr) : _node(current) {} |
|
||||||
Iterator(const Iterator& i) : _node(i._node) {} |
|
||||||
Iterator& operator ++() { _node = _node->next; return *this; } |
|
||||||
bool operator != (const Iterator& i) const { return _node != i._node; } |
|
||||||
const T& operator * () const { return _node->value(); } |
|
||||||
const T* operator -> () const { return &_node->value(); } |
|
||||||
}; |
|
||||||
|
|
||||||
public: |
|
||||||
typedef const Iterator ConstIterator; |
|
||||||
ConstIterator begin() const { return ConstIterator(_root); } |
|
||||||
ConstIterator end() const { return ConstIterator(nullptr); } |
|
||||||
|
|
||||||
LinkedList(OnRemove onRemove) : _root(nullptr), _onRemove(onRemove) {} |
|
||||||
~LinkedList(){} |
|
||||||
void add(const T& t){ |
|
||||||
auto it = new ItemType(t); |
|
||||||
if(!_root){ |
|
||||||
_root = it; |
|
||||||
} else { |
|
||||||
auto i = _root; |
|
||||||
while(i->next) i = i->next; |
|
||||||
i->next = it; |
|
||||||
} |
|
||||||
} |
|
||||||
T& front() const { |
|
||||||
return _root->value(); |
|
||||||
} |
|
||||||
|
|
||||||
bool isEmpty() const { |
|
||||||
return _root == nullptr; |
|
||||||
} |
|
||||||
size_t length() const { |
|
||||||
size_t i = 0; |
|
||||||
auto it = _root; |
|
||||||
while(it){ |
|
||||||
i++; |
|
||||||
it = it->next; |
|
||||||
} |
|
||||||
return i; |
|
||||||
} |
|
||||||
size_t count_if(Predicate predicate) const { |
|
||||||
size_t i = 0; |
|
||||||
auto it = _root; |
|
||||||
while(it){ |
|
||||||
if (!predicate){ |
|
||||||
i++; |
|
||||||
} |
|
||||||
else if (predicate(it->value())) { |
|
||||||
i++; |
|
||||||
} |
|
||||||
it = it->next; |
|
||||||
} |
|
||||||
return i; |
|
||||||
} |
|
||||||
const T* nth(size_t N) const { |
|
||||||
size_t i = 0; |
|
||||||
auto it = _root; |
|
||||||
while(it){ |
|
||||||
if(i++ == N) |
|
||||||
return &(it->value()); |
|
||||||
it = it->next; |
|
||||||
} |
|
||||||
return nullptr; |
|
||||||
} |
|
||||||
bool remove(const T& t){ |
|
||||||
auto it = _root; |
|
||||||
auto pit = _root; |
|
||||||
while(it){ |
|
||||||
if(it->value() == t){ |
|
||||||
if(it == _root){ |
|
||||||
_root = _root->next; |
|
||||||
} else { |
|
||||||
pit->next = it->next; |
|
||||||
} |
|
||||||
|
|
||||||
if (_onRemove) { |
|
||||||
_onRemove(it->value()); |
|
||||||
} |
|
||||||
|
|
||||||
delete it; |
|
||||||
return true; |
|
||||||
} |
|
||||||
pit = it; |
|
||||||
it = it->next; |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
bool remove_first(Predicate predicate){ |
|
||||||
auto it = _root; |
|
||||||
auto pit = _root; |
|
||||||
while(it){ |
|
||||||
if(predicate(it->value())){ |
|
||||||
if(it == _root){ |
|
||||||
_root = _root->next; |
|
||||||
} else { |
|
||||||
pit->next = it->next; |
|
||||||
} |
|
||||||
if (_onRemove) { |
|
||||||
_onRemove(it->value()); |
|
||||||
} |
|
||||||
delete it; |
|
||||||
return true; |
|
||||||
} |
|
||||||
pit = it; |
|
||||||
it = it->next; |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
void free(){ |
|
||||||
while(_root != nullptr){ |
|
||||||
auto it = _root; |
|
||||||
_root = _root->next; |
|
||||||
if (_onRemove) { |
|
||||||
_onRemove(it->value()); |
|
||||||
} |
|
||||||
delete it; |
|
||||||
} |
|
||||||
_root = nullptr; |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
|
|
||||||
class StringArray : public LinkedList<String> { |
|
||||||
public: |
|
||||||
|
|
||||||
StringArray() : LinkedList(nullptr) {} |
|
||||||
|
|
||||||
bool containsIgnoreCase(const String& str){ |
|
||||||
for (const auto& s : *this) { |
|
||||||
if (str.equalsIgnoreCase(s)) { |
|
||||||
return true; |
|
||||||
} |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#endif /* STRINGARRAY_H_ */ |
|
@ -1,235 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#include "WebAuthentication.h" |
|
||||||
#include <libb64/cencode.h> |
|
||||||
#ifdef ESP32 |
|
||||||
#include "mbedtls/md5.h" |
|
||||||
#else |
|
||||||
#include "md5.h" |
|
||||||
#endif |
|
||||||
|
|
||||||
|
|
||||||
// Basic Auth hash = base64("username:password")
|
|
||||||
|
|
||||||
bool checkBasicAuthentication(const char * hash, const char * username, const char * password){ |
|
||||||
if(username == NULL || password == NULL || hash == NULL) |
|
||||||
return false; |
|
||||||
|
|
||||||
size_t toencodeLen = strlen(username)+strlen(password)+1; |
|
||||||
size_t encodedLen = base64_encode_expected_len(toencodeLen); |
|
||||||
if(strlen(hash) != encodedLen) |
|
||||||
return false; |
|
||||||
|
|
||||||
char *toencode = new char[toencodeLen+1]; |
|
||||||
if(toencode == NULL){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
char *encoded = new char[base64_encode_expected_len(toencodeLen)+1]; |
|
||||||
if(encoded == NULL){ |
|
||||||
delete[] toencode; |
|
||||||
return false; |
|
||||||
} |
|
||||||
sprintf(toencode, "%s:%s", username, password); |
|
||||||
if(base64_encode_chars(toencode, toencodeLen, encoded) > 0 && memcmp(hash, encoded, encodedLen) == 0){ |
|
||||||
delete[] toencode; |
|
||||||
delete[] encoded; |
|
||||||
return true; |
|
||||||
} |
|
||||||
delete[] toencode; |
|
||||||
delete[] encoded; |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
static bool getMD5(uint8_t * data, uint16_t len, char * output){//33 bytes or more
|
|
||||||
#ifdef ESP32 |
|
||||||
mbedtls_md5_context _ctx; |
|
||||||
#else |
|
||||||
md5_context_t _ctx; |
|
||||||
#endif |
|
||||||
uint8_t i; |
|
||||||
uint8_t * _buf = (uint8_t*)malloc(16); |
|
||||||
if(_buf == NULL) |
|
||||||
return false; |
|
||||||
memset(_buf, 0x00, 16); |
|
||||||
#ifdef ESP32 |
|
||||||
mbedtls_md5_init(&_ctx); |
|
||||||
mbedtls_md5_starts_ret(&_ctx); |
|
||||||
mbedtls_md5_update_ret(&_ctx, data, len); |
|
||||||
mbedtls_md5_finish_ret(&_ctx, _buf); |
|
||||||
#else |
|
||||||
MD5Init(&_ctx); |
|
||||||
MD5Update(&_ctx, data, len); |
|
||||||
MD5Final(_buf, &_ctx); |
|
||||||
#endif |
|
||||||
for(i = 0; i < 16; i++) { |
|
||||||
sprintf(output + (i * 2), "%02x", _buf[i]); |
|
||||||
} |
|
||||||
free(_buf); |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
static String genRandomMD5(){ |
|
||||||
#ifdef ESP8266 |
|
||||||
uint32_t r = RANDOM_REG32; |
|
||||||
#else |
|
||||||
uint32_t r = rand(); |
|
||||||
#endif |
|
||||||
char * out = (char*)malloc(33); |
|
||||||
if(out == NULL || !getMD5((uint8_t*)(&r), 4, out)) |
|
||||||
return ""; |
|
||||||
String res = String(out); |
|
||||||
free(out); |
|
||||||
return res; |
|
||||||
} |
|
||||||
|
|
||||||
static String stringMD5(const String& in){ |
|
||||||
char * out = (char*)malloc(33); |
|
||||||
if(out == NULL || !getMD5((uint8_t*)(in.c_str()), in.length(), out)) |
|
||||||
return ""; |
|
||||||
String res = String(out); |
|
||||||
free(out); |
|
||||||
return res; |
|
||||||
} |
|
||||||
|
|
||||||
String generateDigestHash(const char * username, const char * password, const char * realm){ |
|
||||||
if(username == NULL || password == NULL || realm == NULL){ |
|
||||||
return ""; |
|
||||||
} |
|
||||||
char * out = (char*)malloc(33); |
|
||||||
String res = String(username); |
|
||||||
res.concat(":"); |
|
||||||
res.concat(realm); |
|
||||||
res.concat(":"); |
|
||||||
String in = res; |
|
||||||
in.concat(password); |
|
||||||
if(out == NULL || !getMD5((uint8_t*)(in.c_str()), in.length(), out)) |
|
||||||
return ""; |
|
||||||
res.concat(out); |
|
||||||
free(out); |
|
||||||
return res; |
|
||||||
} |
|
||||||
|
|
||||||
String requestDigestAuthentication(const char * realm){ |
|
||||||
String header = "realm=\""; |
|
||||||
if(realm == NULL) |
|
||||||
header.concat("asyncesp"); |
|
||||||
else |
|
||||||
header.concat(realm); |
|
||||||
header.concat( "\", qop=\"auth\", nonce=\""); |
|
||||||
header.concat(genRandomMD5()); |
|
||||||
header.concat("\", opaque=\""); |
|
||||||
header.concat(genRandomMD5()); |
|
||||||
header.concat("\""); |
|
||||||
return header; |
|
||||||
} |
|
||||||
|
|
||||||
bool checkDigestAuthentication(const char * header, const char * method, const char * username, const char * password, const char * realm, bool passwordIsHash, const char * nonce, const char * opaque, const char * uri){ |
|
||||||
if(username == NULL || password == NULL || header == NULL || method == NULL){ |
|
||||||
//os_printf("AUTH FAIL: missing requred fields\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
String myHeader = String(header); |
|
||||||
int nextBreak = myHeader.indexOf(","); |
|
||||||
if(nextBreak < 0){ |
|
||||||
//os_printf("AUTH FAIL: no variables\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
String myUsername = String(); |
|
||||||
String myRealm = String(); |
|
||||||
String myNonce = String(); |
|
||||||
String myUri = String(); |
|
||||||
String myResponse = String(); |
|
||||||
String myQop = String(); |
|
||||||
String myNc = String(); |
|
||||||
String myCnonce = String(); |
|
||||||
|
|
||||||
myHeader += ", "; |
|
||||||
do { |
|
||||||
String avLine = myHeader.substring(0, nextBreak); |
|
||||||
avLine.trim(); |
|
||||||
myHeader = myHeader.substring(nextBreak+1); |
|
||||||
nextBreak = myHeader.indexOf(","); |
|
||||||
|
|
||||||
int eqSign = avLine.indexOf("="); |
|
||||||
if(eqSign < 0){ |
|
||||||
//os_printf("AUTH FAIL: no = sign\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
String varName = avLine.substring(0, eqSign); |
|
||||||
avLine = avLine.substring(eqSign + 1); |
|
||||||
if(avLine.startsWith("\"")){ |
|
||||||
avLine = avLine.substring(1, avLine.length() - 1); |
|
||||||
} |
|
||||||
|
|
||||||
if(varName.equals("username")){ |
|
||||||
if(!avLine.equals(username)){ |
|
||||||
//os_printf("AUTH FAIL: username\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
myUsername = avLine; |
|
||||||
} else if(varName.equals("realm")){ |
|
||||||
if(realm != NULL && !avLine.equals(realm)){ |
|
||||||
//os_printf("AUTH FAIL: realm\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
myRealm = avLine; |
|
||||||
} else if(varName.equals("nonce")){ |
|
||||||
if(nonce != NULL && !avLine.equals(nonce)){ |
|
||||||
//os_printf("AUTH FAIL: nonce\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
myNonce = avLine; |
|
||||||
} else if(varName.equals("opaque")){ |
|
||||||
if(opaque != NULL && !avLine.equals(opaque)){ |
|
||||||
//os_printf("AUTH FAIL: opaque\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
} else if(varName.equals("uri")){ |
|
||||||
if(uri != NULL && !avLine.equals(uri)){ |
|
||||||
//os_printf("AUTH FAIL: uri\n");
|
|
||||||
return false; |
|
||||||
} |
|
||||||
myUri = avLine; |
|
||||||
} else if(varName.equals("response")){ |
|
||||||
myResponse = avLine; |
|
||||||
} else if(varName.equals("qop")){ |
|
||||||
myQop = avLine; |
|
||||||
} else if(varName.equals("nc")){ |
|
||||||
myNc = avLine; |
|
||||||
} else if(varName.equals("cnonce")){ |
|
||||||
myCnonce = avLine; |
|
||||||
} |
|
||||||
} while(nextBreak > 0); |
|
||||||
|
|
||||||
String ha1 = (passwordIsHash) ? String(password) : stringMD5(myUsername + ":" + myRealm + ":" + String(password)); |
|
||||||
String ha2 = String(method) + ":" + myUri; |
|
||||||
String response = ha1 + ":" + myNonce + ":" + myNc + ":" + myCnonce + ":" + myQop + ":" + stringMD5(ha2); |
|
||||||
|
|
||||||
if(myResponse.equals(stringMD5(response))){ |
|
||||||
//os_printf("AUTH SUCCESS\n");
|
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
//os_printf("AUTH FAIL: password\n");
|
|
||||||
return false; |
|
||||||
} |
|
@ -1,34 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
|
|
||||||
#ifndef WEB_AUTHENTICATION_H_ |
|
||||||
#define WEB_AUTHENTICATION_H_ |
|
||||||
|
|
||||||
#include "Arduino.h" |
|
||||||
|
|
||||||
bool checkBasicAuthentication(const char * header, const char * username, const char * password); |
|
||||||
String requestDigestAuthentication(const char * realm); |
|
||||||
bool checkDigestAuthentication(const char * header, const char * method, const char * username, const char * password, const char * realm, bool passwordIsHash, const char * nonce, const char * opaque, const char * uri); |
|
||||||
|
|
||||||
//for storing hashed versions on the device that can be authenticated against
|
|
||||||
String generateDigestHash(const char * username, const char * password, const char * realm); |
|
||||||
|
|
||||||
#endif |
|
@ -1,151 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#ifndef ASYNCWEBSERVERHANDLERIMPL_H_ |
|
||||||
#define ASYNCWEBSERVERHANDLERIMPL_H_ |
|
||||||
|
|
||||||
#include <string> |
|
||||||
#ifdef ASYNCWEBSERVER_REGEX |
|
||||||
#include <regex> |
|
||||||
#endif |
|
||||||
|
|
||||||
#include "stddef.h" |
|
||||||
#include <time.h> |
|
||||||
|
|
||||||
class AsyncStaticWebHandler: public AsyncWebHandler { |
|
||||||
using File = fs::File; |
|
||||||
using FS = fs::FS; |
|
||||||
private: |
|
||||||
bool _getFile(AsyncWebServerRequest *request); |
|
||||||
bool _fileExists(AsyncWebServerRequest *request, const String& path); |
|
||||||
uint8_t _countBits(const uint8_t value) const; |
|
||||||
protected: |
|
||||||
FS _fs; |
|
||||||
String _uri; |
|
||||||
String _path; |
|
||||||
String _default_file; |
|
||||||
String _cache_control; |
|
||||||
String _last_modified; |
|
||||||
AwsTemplateProcessor _callback; |
|
||||||
bool _isDir; |
|
||||||
bool _gzipFirst; |
|
||||||
uint8_t _gzipStats; |
|
||||||
public: |
|
||||||
AsyncStaticWebHandler(const char* uri, FS& fs, const char* path, const char* cache_control); |
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request) override final; |
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request) override final; |
|
||||||
AsyncStaticWebHandler& setIsDir(bool isDir); |
|
||||||
AsyncStaticWebHandler& setDefaultFile(const char* filename); |
|
||||||
AsyncStaticWebHandler& setCacheControl(const char* cache_control); |
|
||||||
AsyncStaticWebHandler& setLastModified(const char* last_modified); |
|
||||||
AsyncStaticWebHandler& setLastModified(struct tm* last_modified); |
|
||||||
#ifdef ESP8266 |
|
||||||
AsyncStaticWebHandler& setLastModified(time_t last_modified); |
|
||||||
AsyncStaticWebHandler& setLastModified(); //sets to current time. Make sure sntp is runing and time is updated
|
|
||||||
#endif |
|
||||||
AsyncStaticWebHandler& setTemplateProcessor(AwsTemplateProcessor newCallback) {_callback = newCallback; return *this;} |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncCallbackWebHandler: public AsyncWebHandler { |
|
||||||
private: |
|
||||||
protected: |
|
||||||
String _uri; |
|
||||||
WebRequestMethodComposite _method; |
|
||||||
ArRequestHandlerFunction _onRequest; |
|
||||||
ArUploadHandlerFunction _onUpload; |
|
||||||
ArBodyHandlerFunction _onBody; |
|
||||||
bool _isRegex; |
|
||||||
public: |
|
||||||
AsyncCallbackWebHandler() : _uri(), _method(HTTP_ANY), _onRequest(NULL), _onUpload(NULL), _onBody(NULL), _isRegex(false) {} |
|
||||||
void setUri(const String& uri){
|
|
||||||
_uri = uri;
|
|
||||||
_isRegex = uri.startsWith("^") && uri.endsWith("$"); |
|
||||||
} |
|
||||||
void setMethod(WebRequestMethodComposite method){ _method = method; } |
|
||||||
void onRequest(ArRequestHandlerFunction fn){ _onRequest = fn; } |
|
||||||
void onUpload(ArUploadHandlerFunction fn){ _onUpload = fn; } |
|
||||||
void onBody(ArBodyHandlerFunction fn){ _onBody = fn; } |
|
||||||
|
|
||||||
virtual bool canHandle(AsyncWebServerRequest *request) override final{ |
|
||||||
|
|
||||||
if(!_onRequest) |
|
||||||
return false; |
|
||||||
|
|
||||||
if(!(_method & request->method())) |
|
||||||
return false; |
|
||||||
|
|
||||||
#ifdef ASYNCWEBSERVER_REGEX |
|
||||||
if (_isRegex) { |
|
||||||
std::regex pattern(_uri.c_str()); |
|
||||||
std::smatch matches; |
|
||||||
std::string s(request->url().c_str()); |
|
||||||
if(std::regex_search(s, matches, pattern)) { |
|
||||||
for (size_t i = 1; i < matches.size(); ++i) { // start from 1
|
|
||||||
request->_addPathParam(matches[i].str().c_str()); |
|
||||||
} |
|
||||||
} else { |
|
||||||
return false; |
|
||||||
} |
|
||||||
} else
|
|
||||||
#endif |
|
||||||
if (_uri.length() && _uri.startsWith("/*.")) { |
|
||||||
String uriTemplate = String (_uri); |
|
||||||
uriTemplate = uriTemplate.substring(uriTemplate.lastIndexOf(".")); |
|
||||||
if (!request->url().endsWith(uriTemplate)) |
|
||||||
return false; |
|
||||||
} |
|
||||||
else |
|
||||||
if (_uri.length() && _uri.endsWith("*")) { |
|
||||||
String uriTemplate = String(_uri); |
|
||||||
uriTemplate = uriTemplate.substring(0, uriTemplate.length() - 1); |
|
||||||
if (!request->url().startsWith(uriTemplate)) |
|
||||||
return false; |
|
||||||
} |
|
||||||
else if(_uri.length() && (_uri != request->url() && !request->url().startsWith(_uri+"/"))) |
|
||||||
return false; |
|
||||||
|
|
||||||
request->addInterestingHeader("ANY"); |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
virtual void handleRequest(AsyncWebServerRequest *request) override final { |
|
||||||
if((_username != "" && _password != "") && !request->authenticate(_username.c_str(), _password.c_str())) |
|
||||||
return request->requestAuthentication(); |
|
||||||
if(_onRequest) |
|
||||||
_onRequest(request); |
|
||||||
else |
|
||||||
request->send(500); |
|
||||||
} |
|
||||||
virtual void handleUpload(AsyncWebServerRequest *request, const String& filename, size_t index, uint8_t *data, size_t len, bool final) override final { |
|
||||||
if((_username != "" && _password != "") && !request->authenticate(_username.c_str(), _password.c_str())) |
|
||||||
return request->requestAuthentication(); |
|
||||||
if(_onUpload) |
|
||||||
_onUpload(request, filename, index, data, len, final); |
|
||||||
} |
|
||||||
virtual void handleBody(AsyncWebServerRequest *request, uint8_t *data, size_t len, size_t index, size_t total) override final { |
|
||||||
if((_username != "" && _password != "") && !request->authenticate(_username.c_str(), _password.c_str())) |
|
||||||
return request->requestAuthentication(); |
|
||||||
if(_onBody) |
|
||||||
_onBody(request, data, len, index, total); |
|
||||||
} |
|
||||||
virtual bool isRequestHandlerTrivial() override final {return _onRequest ? false : true;} |
|
||||||
}; |
|
||||||
|
|
||||||
#endif /* ASYNCWEBSERVERHANDLERIMPL_H_ */ |
|
@ -1,220 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#include "ESPAsyncWebServer.h" |
|
||||||
#include "WebHandlerImpl.h" |
|
||||||
|
|
||||||
AsyncStaticWebHandler::AsyncStaticWebHandler(const char* uri, FS& fs, const char* path, const char* cache_control) |
|
||||||
: _fs(fs), _uri(uri), _path(path), _default_file("index.htm"), _cache_control(cache_control), _last_modified(""), _callback(nullptr) |
|
||||||
{ |
|
||||||
// Ensure leading '/'
|
|
||||||
if (_uri.length() == 0 || _uri[0] != '/') _uri = "/" + _uri; |
|
||||||
if (_path.length() == 0 || _path[0] != '/') _path = "/" + _path; |
|
||||||
|
|
||||||
// If path ends with '/' we assume a hint that this is a directory to improve performance.
|
|
||||||
// However - if it does not end with '/' we, can't assume a file, path can still be a directory.
|
|
||||||
_isDir = _path[_path.length()-1] == '/'; |
|
||||||
|
|
||||||
// Remove the trailing '/' so we can handle default file
|
|
||||||
// Notice that root will be "" not "/"
|
|
||||||
if (_uri[_uri.length()-1] == '/') _uri = _uri.substring(0, _uri.length()-1); |
|
||||||
if (_path[_path.length()-1] == '/') _path = _path.substring(0, _path.length()-1); |
|
||||||
|
|
||||||
// Reset stats
|
|
||||||
_gzipFirst = false; |
|
||||||
_gzipStats = 0xF8; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setIsDir(bool isDir){ |
|
||||||
_isDir = isDir; |
|
||||||
return *this; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setDefaultFile(const char* filename){ |
|
||||||
_default_file = String(filename); |
|
||||||
return *this; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setCacheControl(const char* cache_control){ |
|
||||||
_cache_control = String(cache_control); |
|
||||||
return *this; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setLastModified(const char* last_modified){ |
|
||||||
_last_modified = String(last_modified); |
|
||||||
return *this; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setLastModified(struct tm* last_modified){ |
|
||||||
char result[30]; |
|
||||||
strftime (result,30,"%a, %d %b %Y %H:%M:%S %Z", last_modified); |
|
||||||
return setLastModified((const char *)result); |
|
||||||
} |
|
||||||
|
|
||||||
#ifdef ESP8266 |
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setLastModified(time_t last_modified){ |
|
||||||
return setLastModified((struct tm *)gmtime(&last_modified)); |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncStaticWebHandler::setLastModified(){ |
|
||||||
time_t last_modified; |
|
||||||
if(time(&last_modified) == 0) //time is not yet set
|
|
||||||
return *this; |
|
||||||
return setLastModified(last_modified); |
|
||||||
} |
|
||||||
#endif |
|
||||||
bool AsyncStaticWebHandler::canHandle(AsyncWebServerRequest *request){ |
|
||||||
if(request->method() != HTTP_GET
|
|
||||||
|| !request->url().startsWith(_uri)
|
|
||||||
|| !request->isExpectedRequestedConnType(RCT_DEFAULT, RCT_HTTP) |
|
||||||
){ |
|
||||||
return false; |
|
||||||
} |
|
||||||
if (_getFile(request)) { |
|
||||||
// We interested in "If-Modified-Since" header to check if file was modified
|
|
||||||
if (_last_modified.length()) |
|
||||||
request->addInterestingHeader("If-Modified-Since"); |
|
||||||
|
|
||||||
if(_cache_control.length()) |
|
||||||
request->addInterestingHeader("If-None-Match"); |
|
||||||
|
|
||||||
DEBUGF("[AsyncStaticWebHandler::canHandle] TRUE\n"); |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
bool AsyncStaticWebHandler::_getFile(AsyncWebServerRequest *request) |
|
||||||
{ |
|
||||||
// Remove the found uri
|
|
||||||
String path = request->url().substring(_uri.length()); |
|
||||||
|
|
||||||
// We can skip the file check and look for default if request is to the root of a directory or that request path ends with '/'
|
|
||||||
bool canSkipFileCheck = (_isDir && path.length() == 0) || (path.length() && path[path.length()-1] == '/'); |
|
||||||
|
|
||||||
path = _path + path; |
|
||||||
|
|
||||||
// Do we have a file or .gz file
|
|
||||||
if (!canSkipFileCheck && _fileExists(request, path)) |
|
||||||
return true; |
|
||||||
|
|
||||||
// Can't handle if not default file
|
|
||||||
if (_default_file.length() == 0) |
|
||||||
return false; |
|
||||||
|
|
||||||
// Try to add default file, ensure there is a trailing '/' ot the path.
|
|
||||||
if (path.length() == 0 || path[path.length()-1] != '/') |
|
||||||
path += "/"; |
|
||||||
path += _default_file; |
|
||||||
|
|
||||||
return _fileExists(request, path); |
|
||||||
} |
|
||||||
|
|
||||||
#ifdef ESP32 |
|
||||||
#define FILE_IS_REAL(f) (f == true && !f.isDirectory()) |
|
||||||
#else |
|
||||||
#define FILE_IS_REAL(f) (f == true) |
|
||||||
#endif |
|
||||||
|
|
||||||
bool AsyncStaticWebHandler::_fileExists(AsyncWebServerRequest *request, const String& path) |
|
||||||
{ |
|
||||||
bool fileFound = false; |
|
||||||
bool gzipFound = false; |
|
||||||
|
|
||||||
String gzip = path + ".gz"; |
|
||||||
|
|
||||||
if (_gzipFirst) { |
|
||||||
request->_tempFile = _fs.open(gzip, "r"); |
|
||||||
gzipFound = FILE_IS_REAL(request->_tempFile); |
|
||||||
if (!gzipFound){ |
|
||||||
request->_tempFile = _fs.open(path, "r"); |
|
||||||
fileFound = FILE_IS_REAL(request->_tempFile); |
|
||||||
} |
|
||||||
} else { |
|
||||||
request->_tempFile = _fs.open(path, "r"); |
|
||||||
fileFound = FILE_IS_REAL(request->_tempFile); |
|
||||||
if (!fileFound){ |
|
||||||
request->_tempFile = _fs.open(gzip, "r"); |
|
||||||
gzipFound = FILE_IS_REAL(request->_tempFile); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
bool found = fileFound || gzipFound; |
|
||||||
|
|
||||||
if (found) { |
|
||||||
// Extract the file name from the path and keep it in _tempObject
|
|
||||||
size_t pathLen = path.length(); |
|
||||||
char * _tempPath = (char*)malloc(pathLen+1); |
|
||||||
snprintf(_tempPath, pathLen+1, "%s", path.c_str()); |
|
||||||
request->_tempObject = (void*)_tempPath; |
|
||||||
|
|
||||||
// Calculate gzip statistic
|
|
||||||
_gzipStats = (_gzipStats << 1) + (gzipFound ? 1 : 0); |
|
||||||
if (_gzipStats == 0x00) _gzipFirst = false; // All files are not gzip
|
|
||||||
else if (_gzipStats == 0xFF) _gzipFirst = true; // All files are gzip
|
|
||||||
else _gzipFirst = _countBits(_gzipStats) > 4; // IF we have more gzip files - try gzip first
|
|
||||||
} |
|
||||||
|
|
||||||
return found; |
|
||||||
} |
|
||||||
|
|
||||||
uint8_t AsyncStaticWebHandler::_countBits(const uint8_t value) const |
|
||||||
{ |
|
||||||
uint8_t w = value; |
|
||||||
uint8_t n; |
|
||||||
for (n=0; w!=0; n++) w&=w-1; |
|
||||||
return n; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncStaticWebHandler::handleRequest(AsyncWebServerRequest *request) |
|
||||||
{ |
|
||||||
// Get the filename from request->_tempObject and free it
|
|
||||||
String filename = String((char*)request->_tempObject); |
|
||||||
free(request->_tempObject); |
|
||||||
request->_tempObject = NULL; |
|
||||||
if((_username != "" && _password != "") && !request->authenticate(_username.c_str(), _password.c_str())) |
|
||||||
return request->requestAuthentication(); |
|
||||||
|
|
||||||
if (request->_tempFile == true) { |
|
||||||
String etag = String(request->_tempFile.size()); |
|
||||||
if (_last_modified.length() && _last_modified == request->header("If-Modified-Since")) { |
|
||||||
request->_tempFile.close(); |
|
||||||
request->send(304); // Not modified
|
|
||||||
} else if (_cache_control.length() && request->hasHeader("If-None-Match") && request->header("If-None-Match").equals(etag)) { |
|
||||||
request->_tempFile.close(); |
|
||||||
AsyncWebServerResponse * response = new AsyncBasicResponse(304); // Not modified
|
|
||||||
response->addHeader("Cache-Control", _cache_control); |
|
||||||
response->addHeader("ETag", etag); |
|
||||||
request->send(response); |
|
||||||
} else { |
|
||||||
AsyncWebServerResponse * response = new AsyncFileResponse(request->_tempFile, filename, String(), false, _callback); |
|
||||||
if (_last_modified.length()) |
|
||||||
response->addHeader("Last-Modified", _last_modified); |
|
||||||
if (_cache_control.length()){ |
|
||||||
response->addHeader("Cache-Control", _cache_control); |
|
||||||
response->addHeader("ETag", etag); |
|
||||||
} |
|
||||||
request->send(response); |
|
||||||
} |
|
||||||
} else { |
|
||||||
request->send(404); |
|
||||||
} |
|
||||||
} |
|
File diff suppressed because it is too large
Load Diff
@ -1,136 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#ifndef ASYNCWEBSERVERRESPONSEIMPL_H_ |
|
||||||
#define ASYNCWEBSERVERRESPONSEIMPL_H_ |
|
||||||
|
|
||||||
#ifdef Arduino_h |
|
||||||
// arduino is not compatible with std::vector
|
|
||||||
#undef min |
|
||||||
#undef max |
|
||||||
#endif |
|
||||||
#include <vector> |
|
||||||
// It is possible to restore these defines, but one can use _min and _max instead. Or std::min, std::max.
|
|
||||||
|
|
||||||
class AsyncBasicResponse: public AsyncWebServerResponse { |
|
||||||
private: |
|
||||||
String _content; |
|
||||||
public: |
|
||||||
AsyncBasicResponse(int code, const String& contentType=String(), const String& content=String()); |
|
||||||
void _respond(AsyncWebServerRequest *request); |
|
||||||
size_t _ack(AsyncWebServerRequest *request, size_t len, uint32_t time); |
|
||||||
bool _sourceValid() const { return true; } |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncAbstractResponse: public AsyncWebServerResponse { |
|
||||||
private: |
|
||||||
String _head; |
|
||||||
// Data is inserted into cache at begin().
|
|
||||||
// This is inefficient with vector, but if we use some other container,
|
|
||||||
// we won't be able to access it as contiguous array of bytes when reading from it,
|
|
||||||
// so by gaining performance in one place, we'll lose it in another.
|
|
||||||
std::vector<uint8_t> _cache; |
|
||||||
size_t _readDataFromCacheOrContent(uint8_t* data, const size_t len); |
|
||||||
size_t _fillBufferAndProcessTemplates(uint8_t* buf, size_t maxLen); |
|
||||||
protected: |
|
||||||
AwsTemplateProcessor _callback; |
|
||||||
public: |
|
||||||
AsyncAbstractResponse(AwsTemplateProcessor callback=nullptr); |
|
||||||
void _respond(AsyncWebServerRequest *request); |
|
||||||
size_t _ack(AsyncWebServerRequest *request, size_t len, uint32_t time); |
|
||||||
bool _sourceValid() const { return false; } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf __attribute__((unused)), size_t maxLen __attribute__((unused))) { return 0; } |
|
||||||
}; |
|
||||||
|
|
||||||
#ifndef TEMPLATE_PLACEHOLDER |
|
||||||
#define TEMPLATE_PLACEHOLDER '%' |
|
||||||
#endif |
|
||||||
|
|
||||||
#define TEMPLATE_PARAM_NAME_LENGTH 32 |
|
||||||
class AsyncFileResponse: public AsyncAbstractResponse { |
|
||||||
using File = fs::File; |
|
||||||
using FS = fs::FS; |
|
||||||
private: |
|
||||||
File _content; |
|
||||||
String _path; |
|
||||||
void _setContentType(const String& path); |
|
||||||
public: |
|
||||||
AsyncFileResponse(FS &fs, const String& path, const String& contentType=String(), bool download=false, AwsTemplateProcessor callback=nullptr); |
|
||||||
AsyncFileResponse(File content, const String& path, const String& contentType=String(), bool download=false, AwsTemplateProcessor callback=nullptr); |
|
||||||
~AsyncFileResponse(); |
|
||||||
bool _sourceValid() const { return !!(_content); } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf, size_t maxLen) override; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncStreamResponse: public AsyncAbstractResponse { |
|
||||||
private: |
|
||||||
Stream *_content; |
|
||||||
public: |
|
||||||
AsyncStreamResponse(Stream &stream, const String& contentType, size_t len, AwsTemplateProcessor callback=nullptr); |
|
||||||
bool _sourceValid() const { return !!(_content); } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf, size_t maxLen) override; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncCallbackResponse: public AsyncAbstractResponse { |
|
||||||
private: |
|
||||||
AwsResponseFiller _content; |
|
||||||
size_t _filledLength; |
|
||||||
public: |
|
||||||
AsyncCallbackResponse(const String& contentType, size_t len, AwsResponseFiller callback, AwsTemplateProcessor templateCallback=nullptr); |
|
||||||
bool _sourceValid() const { return !!(_content); } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf, size_t maxLen) override; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncChunkedResponse: public AsyncAbstractResponse { |
|
||||||
private: |
|
||||||
AwsResponseFiller _content; |
|
||||||
size_t _filledLength; |
|
||||||
public: |
|
||||||
AsyncChunkedResponse(const String& contentType, AwsResponseFiller callback, AwsTemplateProcessor templateCallback=nullptr); |
|
||||||
bool _sourceValid() const { return !!(_content); } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf, size_t maxLen) override; |
|
||||||
}; |
|
||||||
|
|
||||||
class AsyncProgmemResponse: public AsyncAbstractResponse { |
|
||||||
private: |
|
||||||
const uint8_t * _content; |
|
||||||
size_t _readLength; |
|
||||||
public: |
|
||||||
AsyncProgmemResponse(int code, const String& contentType, const uint8_t * content, size_t len, AwsTemplateProcessor callback=nullptr); |
|
||||||
bool _sourceValid() const { return true; } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf, size_t maxLen) override; |
|
||||||
}; |
|
||||||
|
|
||||||
class cbuf; |
|
||||||
|
|
||||||
class AsyncResponseStream: public AsyncAbstractResponse, public Print { |
|
||||||
private: |
|
||||||
cbuf *_content; |
|
||||||
public: |
|
||||||
AsyncResponseStream(const String& contentType, size_t bufferSize); |
|
||||||
~AsyncResponseStream(); |
|
||||||
bool _sourceValid() const { return (_state < RESPONSE_END); } |
|
||||||
virtual size_t _fillBuffer(uint8_t *buf, size_t maxLen) override; |
|
||||||
size_t write(const uint8_t *data, size_t len); |
|
||||||
size_t write(uint8_t data); |
|
||||||
using Print::write; |
|
||||||
}; |
|
||||||
|
|
||||||
#endif /* ASYNCWEBSERVERRESPONSEIMPL_H_ */ |
|
@ -1,699 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#include "ESPAsyncWebServer.h" |
|
||||||
#include "WebResponseImpl.h" |
|
||||||
#include "cbuf.h" |
|
||||||
|
|
||||||
// Since ESP8266 does not link memchr by default, here's its implementation.
|
|
||||||
void* memchr(void* ptr, int ch, size_t count) |
|
||||||
{ |
|
||||||
unsigned char* p = static_cast<unsigned char*>(ptr); |
|
||||||
while(count--) |
|
||||||
if(*p++ == static_cast<unsigned char>(ch)) |
|
||||||
return --p; |
|
||||||
return nullptr; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Abstract Response |
|
||||||
* */ |
|
||||||
const char* AsyncWebServerResponse::_responseCodeToString(int code) { |
|
||||||
switch (code) { |
|
||||||
case 100: return "Continue"; |
|
||||||
case 101: return "Switching Protocols"; |
|
||||||
case 200: return "OK"; |
|
||||||
case 201: return "Created"; |
|
||||||
case 202: return "Accepted"; |
|
||||||
case 203: return "Non-Authoritative Information"; |
|
||||||
case 204: return "No Content"; |
|
||||||
case 205: return "Reset Content"; |
|
||||||
case 206: return "Partial Content"; |
|
||||||
case 300: return "Multiple Choices"; |
|
||||||
case 301: return "Moved Permanently"; |
|
||||||
case 302: return "Found"; |
|
||||||
case 303: return "See Other"; |
|
||||||
case 304: return "Not Modified"; |
|
||||||
case 305: return "Use Proxy"; |
|
||||||
case 307: return "Temporary Redirect"; |
|
||||||
case 400: return "Bad Request"; |
|
||||||
case 401: return "Unauthorized"; |
|
||||||
case 402: return "Payment Required"; |
|
||||||
case 403: return "Forbidden"; |
|
||||||
case 404: return "Not Found"; |
|
||||||
case 405: return "Method Not Allowed"; |
|
||||||
case 406: return "Not Acceptable"; |
|
||||||
case 407: return "Proxy Authentication Required"; |
|
||||||
case 408: return "Request Time-out"; |
|
||||||
case 409: return "Conflict"; |
|
||||||
case 410: return "Gone"; |
|
||||||
case 411: return "Length Required"; |
|
||||||
case 412: return "Precondition Failed"; |
|
||||||
case 413: return "Request Entity Too Large"; |
|
||||||
case 414: return "Request-URI Too Large"; |
|
||||||
case 415: return "Unsupported Media Type"; |
|
||||||
case 416: return "Requested range not satisfiable"; |
|
||||||
case 417: return "Expectation Failed"; |
|
||||||
case 500: return "Internal Server Error"; |
|
||||||
case 501: return "Not Implemented"; |
|
||||||
case 502: return "Bad Gateway"; |
|
||||||
case 503: return "Service Unavailable"; |
|
||||||
case 504: return "Gateway Time-out"; |
|
||||||
case 505: return "HTTP Version not supported"; |
|
||||||
default: return ""; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
AsyncWebServerResponse::AsyncWebServerResponse() |
|
||||||
: _code(0) |
|
||||||
, _headers(LinkedList<AsyncWebHeader *>([](AsyncWebHeader *h){ delete h; })) |
|
||||||
, _contentType() |
|
||||||
, _contentLength(0) |
|
||||||
, _sendContentLength(true) |
|
||||||
, _chunked(false) |
|
||||||
, _headLength(0) |
|
||||||
, _sentLength(0) |
|
||||||
, _ackedLength(0) |
|
||||||
, _writtenLength(0) |
|
||||||
, _state(RESPONSE_SETUP) |
|
||||||
{ |
|
||||||
for(auto header: DefaultHeaders::Instance()) { |
|
||||||
_headers.add(new AsyncWebHeader(header->name(), header->value())); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
AsyncWebServerResponse::~AsyncWebServerResponse(){ |
|
||||||
_headers.free(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServerResponse::setCode(int code){ |
|
||||||
if(_state == RESPONSE_SETUP) |
|
||||||
_code = code; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServerResponse::setContentLength(size_t len){ |
|
||||||
if(_state == RESPONSE_SETUP) |
|
||||||
_contentLength = len; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServerResponse::setContentType(const String& type){ |
|
||||||
if(_state == RESPONSE_SETUP) |
|
||||||
_contentType = type; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServerResponse::addHeader(const String& name, const String& value){ |
|
||||||
_headers.add(new AsyncWebHeader(name, value)); |
|
||||||
} |
|
||||||
|
|
||||||
String AsyncWebServerResponse::_assembleHead(uint8_t version){ |
|
||||||
if(version){ |
|
||||||
addHeader("Accept-Ranges","none"); |
|
||||||
if(_chunked) |
|
||||||
addHeader("Transfer-Encoding","chunked"); |
|
||||||
} |
|
||||||
String out = String(); |
|
||||||
int bufSize = 300; |
|
||||||
char buf[bufSize]; |
|
||||||
|
|
||||||
snprintf(buf, bufSize, "HTTP/1.%d %d %s\r\n", version, _code, _responseCodeToString(_code)); |
|
||||||
out.concat(buf); |
|
||||||
|
|
||||||
if(_sendContentLength) { |
|
||||||
snprintf(buf, bufSize, "Content-Length: %d\r\n", _contentLength); |
|
||||||
out.concat(buf); |
|
||||||
} |
|
||||||
if(_contentType.length()) { |
|
||||||
snprintf(buf, bufSize, "Content-Type: %s\r\n", _contentType.c_str()); |
|
||||||
out.concat(buf); |
|
||||||
} |
|
||||||
|
|
||||||
for(const auto& header: _headers){ |
|
||||||
snprintf(buf, bufSize, "%s: %s\r\n", header->name().c_str(), header->value().c_str()); |
|
||||||
out.concat(buf); |
|
||||||
} |
|
||||||
_headers.free(); |
|
||||||
|
|
||||||
out.concat("\r\n"); |
|
||||||
_headLength = out.length(); |
|
||||||
return out; |
|
||||||
} |
|
||||||
|
|
||||||
bool AsyncWebServerResponse::_started() const { return _state > RESPONSE_SETUP; } |
|
||||||
bool AsyncWebServerResponse::_finished() const { return _state > RESPONSE_WAIT_ACK; } |
|
||||||
bool AsyncWebServerResponse::_failed() const { return _state == RESPONSE_FAILED; } |
|
||||||
bool AsyncWebServerResponse::_sourceValid() const { return false; } |
|
||||||
void AsyncWebServerResponse::_respond(AsyncWebServerRequest *request){ _state = RESPONSE_END; request->client()->close(); } |
|
||||||
size_t AsyncWebServerResponse::_ack(AsyncWebServerRequest *request, size_t len, uint32_t time){ (void)request; (void)len; (void)time; return 0; } |
|
||||||
|
|
||||||
/*
|
|
||||||
* String/Code Response |
|
||||||
* */ |
|
||||||
AsyncBasicResponse::AsyncBasicResponse(int code, const String& contentType, const String& content){ |
|
||||||
_code = code; |
|
||||||
_content = content; |
|
||||||
_contentType = contentType; |
|
||||||
if(_content.length()){ |
|
||||||
_contentLength = _content.length(); |
|
||||||
if(!_contentType.length()) |
|
||||||
_contentType = "text/plain"; |
|
||||||
} |
|
||||||
addHeader("Connection","close"); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncBasicResponse::_respond(AsyncWebServerRequest *request){ |
|
||||||
_state = RESPONSE_HEADERS; |
|
||||||
String out = _assembleHead(request->version()); |
|
||||||
size_t outLen = out.length(); |
|
||||||
size_t space = request->client()->space(); |
|
||||||
if(!_contentLength && space >= outLen){ |
|
||||||
_writtenLength += request->client()->write(out.c_str(), outLen); |
|
||||||
_state = RESPONSE_WAIT_ACK; |
|
||||||
} else if(_contentLength && space >= outLen + _contentLength){ |
|
||||||
out += _content; |
|
||||||
outLen += _contentLength; |
|
||||||
_writtenLength += request->client()->write(out.c_str(), outLen); |
|
||||||
_state = RESPONSE_WAIT_ACK; |
|
||||||
} else if(space && space < outLen){ |
|
||||||
String partial = out.substring(0, space); |
|
||||||
_content = out.substring(space) + _content; |
|
||||||
_contentLength += outLen - space; |
|
||||||
_writtenLength += request->client()->write(partial.c_str(), partial.length()); |
|
||||||
_state = RESPONSE_CONTENT; |
|
||||||
} else if(space > outLen && space < (outLen + _contentLength)){ |
|
||||||
size_t shift = space - outLen; |
|
||||||
outLen += shift; |
|
||||||
_sentLength += shift; |
|
||||||
out += _content.substring(0, shift); |
|
||||||
_content = _content.substring(shift); |
|
||||||
_writtenLength += request->client()->write(out.c_str(), outLen); |
|
||||||
_state = RESPONSE_CONTENT; |
|
||||||
} else { |
|
||||||
_content = out + _content; |
|
||||||
_contentLength += outLen; |
|
||||||
_state = RESPONSE_CONTENT; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncBasicResponse::_ack(AsyncWebServerRequest *request, size_t len, uint32_t time){ |
|
||||||
(void)time; |
|
||||||
_ackedLength += len; |
|
||||||
if(_state == RESPONSE_CONTENT){ |
|
||||||
size_t available = _contentLength - _sentLength; |
|
||||||
size_t space = request->client()->space(); |
|
||||||
//we can fit in this packet
|
|
||||||
if(space > available){ |
|
||||||
_writtenLength += request->client()->write(_content.c_str(), available); |
|
||||||
_content = String(); |
|
||||||
_state = RESPONSE_WAIT_ACK; |
|
||||||
return available; |
|
||||||
} |
|
||||||
//send some data, the rest on ack
|
|
||||||
String out = _content.substring(0, space); |
|
||||||
_content = _content.substring(space); |
|
||||||
_sentLength += space; |
|
||||||
_writtenLength += request->client()->write(out.c_str(), space); |
|
||||||
return space; |
|
||||||
} else if(_state == RESPONSE_WAIT_ACK){ |
|
||||||
if(_ackedLength >= _writtenLength){ |
|
||||||
_state = RESPONSE_END; |
|
||||||
} |
|
||||||
} |
|
||||||
return 0; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Abstract Response |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncAbstractResponse::AsyncAbstractResponse(AwsTemplateProcessor callback): _callback(callback) |
|
||||||
{ |
|
||||||
// In case of template processing, we're unable to determine real response size
|
|
||||||
if(callback) { |
|
||||||
_contentLength = 0; |
|
||||||
_sendContentLength = false; |
|
||||||
_chunked = true; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncAbstractResponse::_respond(AsyncWebServerRequest *request){ |
|
||||||
addHeader("Connection","close"); |
|
||||||
_head = _assembleHead(request->version()); |
|
||||||
_state = RESPONSE_HEADERS; |
|
||||||
_ack(request, 0, 0); |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncAbstractResponse::_ack(AsyncWebServerRequest *request, size_t len, uint32_t time){ |
|
||||||
(void)time; |
|
||||||
if(!_sourceValid()){ |
|
||||||
_state = RESPONSE_FAILED; |
|
||||||
request->client()->close(); |
|
||||||
return 0; |
|
||||||
} |
|
||||||
_ackedLength += len; |
|
||||||
size_t space = request->client()->space(); |
|
||||||
|
|
||||||
size_t headLen = _head.length(); |
|
||||||
if(_state == RESPONSE_HEADERS){ |
|
||||||
if(space >= headLen){ |
|
||||||
_state = RESPONSE_CONTENT; |
|
||||||
space -= headLen; |
|
||||||
} else { |
|
||||||
String out = _head.substring(0, space); |
|
||||||
_head = _head.substring(space); |
|
||||||
_writtenLength += request->client()->write(out.c_str(), out.length()); |
|
||||||
return out.length(); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
if(_state == RESPONSE_CONTENT){ |
|
||||||
size_t outLen; |
|
||||||
if(_chunked){ |
|
||||||
if(space <= 8){ |
|
||||||
return 0; |
|
||||||
} |
|
||||||
outLen = space; |
|
||||||
} else if(!_sendContentLength){ |
|
||||||
outLen = space; |
|
||||||
} else { |
|
||||||
outLen = ((_contentLength - _sentLength) > space)?space:(_contentLength - _sentLength); |
|
||||||
} |
|
||||||
|
|
||||||
uint8_t *buf = (uint8_t *)malloc(outLen+headLen); |
|
||||||
if (!buf) { |
|
||||||
// os_printf("_ack malloc %d failed\n", outLen+headLen);
|
|
||||||
return 0; |
|
||||||
} |
|
||||||
|
|
||||||
if(headLen){ |
|
||||||
memcpy(buf, _head.c_str(), _head.length()); |
|
||||||
} |
|
||||||
|
|
||||||
size_t readLen = 0; |
|
||||||
|
|
||||||
if(_chunked){ |
|
||||||
// HTTP 1.1 allows leading zeros in chunk length. Or spaces may be added.
|
|
||||||
// See RFC2616 sections 2, 3.6.1.
|
|
||||||
readLen = _fillBufferAndProcessTemplates(buf+headLen+6, outLen - 8); |
|
||||||
if(readLen == RESPONSE_TRY_AGAIN){ |
|
||||||
free(buf); |
|
||||||
return 0; |
|
||||||
} |
|
||||||
outLen = sprintf((char*)buf+headLen, "%x", readLen) + headLen; |
|
||||||
while(outLen < headLen + 4) buf[outLen++] = ' '; |
|
||||||
buf[outLen++] = '\r'; |
|
||||||
buf[outLen++] = '\n'; |
|
||||||
outLen += readLen; |
|
||||||
buf[outLen++] = '\r'; |
|
||||||
buf[outLen++] = '\n'; |
|
||||||
} else { |
|
||||||
readLen = _fillBufferAndProcessTemplates(buf+headLen, outLen); |
|
||||||
if(readLen == RESPONSE_TRY_AGAIN){ |
|
||||||
free(buf); |
|
||||||
return 0; |
|
||||||
} |
|
||||||
outLen = readLen + headLen; |
|
||||||
} |
|
||||||
|
|
||||||
if(headLen){ |
|
||||||
_head = String(); |
|
||||||
} |
|
||||||
|
|
||||||
if(outLen){ |
|
||||||
_writtenLength += request->client()->write((const char*)buf, outLen); |
|
||||||
} |
|
||||||
|
|
||||||
if(_chunked){ |
|
||||||
_sentLength += readLen; |
|
||||||
} else { |
|
||||||
_sentLength += outLen - headLen; |
|
||||||
} |
|
||||||
|
|
||||||
free(buf); |
|
||||||
|
|
||||||
if((_chunked && readLen == 0) || (!_sendContentLength && outLen == 0) || (!_chunked && _sentLength == _contentLength)){ |
|
||||||
_state = RESPONSE_WAIT_ACK; |
|
||||||
} |
|
||||||
return outLen; |
|
||||||
|
|
||||||
} else if(_state == RESPONSE_WAIT_ACK){ |
|
||||||
if(!_sendContentLength || _ackedLength >= _writtenLength){ |
|
||||||
_state = RESPONSE_END; |
|
||||||
if(!_chunked && !_sendContentLength) |
|
||||||
request->client()->close(true); |
|
||||||
} |
|
||||||
} |
|
||||||
return 0; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncAbstractResponse::_readDataFromCacheOrContent(uint8_t* data, const size_t len) |
|
||||||
{ |
|
||||||
// If we have something in cache, copy it to buffer
|
|
||||||
const size_t readFromCache = std::min(len, _cache.size()); |
|
||||||
if(readFromCache) { |
|
||||||
memcpy(data, _cache.data(), readFromCache); |
|
||||||
_cache.erase(_cache.begin(), _cache.begin() + readFromCache); |
|
||||||
} |
|
||||||
// If we need to read more...
|
|
||||||
const size_t needFromFile = len - readFromCache; |
|
||||||
const size_t readFromContent = _fillBuffer(data + readFromCache, needFromFile); |
|
||||||
return readFromCache + readFromContent; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncAbstractResponse::_fillBufferAndProcessTemplates(uint8_t* data, size_t len) |
|
||||||
{ |
|
||||||
if(!_callback) |
|
||||||
return _fillBuffer(data, len); |
|
||||||
|
|
||||||
const size_t originalLen = len; |
|
||||||
len = _readDataFromCacheOrContent(data, len); |
|
||||||
// Now we've read 'len' bytes, either from cache or from file
|
|
||||||
// Search for template placeholders
|
|
||||||
uint8_t* pTemplateStart = data; |
|
||||||
while((pTemplateStart < &data[len]) && (pTemplateStart = (uint8_t*)memchr(pTemplateStart, TEMPLATE_PLACEHOLDER, &data[len - 1] - pTemplateStart + 1))) { // data[0] ... data[len - 1]
|
|
||||||
uint8_t* pTemplateEnd = (pTemplateStart < &data[len - 1]) ? (uint8_t*)memchr(pTemplateStart + 1, TEMPLATE_PLACEHOLDER, &data[len - 1] - pTemplateStart) : nullptr; |
|
||||||
// temporary buffer to hold parameter name
|
|
||||||
uint8_t buf[TEMPLATE_PARAM_NAME_LENGTH + 1]; |
|
||||||
String paramName; |
|
||||||
// If closing placeholder is found:
|
|
||||||
if(pTemplateEnd) { |
|
||||||
// prepare argument to callback
|
|
||||||
const size_t paramNameLength = std::min(sizeof(buf) - 1, (unsigned int)(pTemplateEnd - pTemplateStart - 1)); |
|
||||||
if(paramNameLength) { |
|
||||||
memcpy(buf, pTemplateStart + 1, paramNameLength); |
|
||||||
buf[paramNameLength] = 0; |
|
||||||
paramName = String(reinterpret_cast<char*>(buf)); |
|
||||||
} else { // double percent sign encountered, this is single percent sign escaped.
|
|
||||||
// remove the 2nd percent sign
|
|
||||||
memmove(pTemplateEnd, pTemplateEnd + 1, &data[len] - pTemplateEnd - 1); |
|
||||||
len += _readDataFromCacheOrContent(&data[len - 1], 1) - 1; |
|
||||||
++pTemplateStart; |
|
||||||
} |
|
||||||
} else if(&data[len - 1] - pTemplateStart + 1 < TEMPLATE_PARAM_NAME_LENGTH + 2) { // closing placeholder not found, check if it's in the remaining file data
|
|
||||||
memcpy(buf, pTemplateStart + 1, &data[len - 1] - pTemplateStart); |
|
||||||
const size_t readFromCacheOrContent = _readDataFromCacheOrContent(buf + (&data[len - 1] - pTemplateStart), TEMPLATE_PARAM_NAME_LENGTH + 2 - (&data[len - 1] - pTemplateStart + 1)); |
|
||||||
if(readFromCacheOrContent) { |
|
||||||
pTemplateEnd = (uint8_t*)memchr(buf + (&data[len - 1] - pTemplateStart), TEMPLATE_PLACEHOLDER, readFromCacheOrContent); |
|
||||||
if(pTemplateEnd) { |
|
||||||
// prepare argument to callback
|
|
||||||
*pTemplateEnd = 0; |
|
||||||
paramName = String(reinterpret_cast<char*>(buf)); |
|
||||||
// Copy remaining read-ahead data into cache
|
|
||||||
_cache.insert(_cache.begin(), pTemplateEnd + 1, buf + (&data[len - 1] - pTemplateStart) + readFromCacheOrContent); |
|
||||||
pTemplateEnd = &data[len - 1]; |
|
||||||
} |
|
||||||
else // closing placeholder not found in file data, store found percent symbol as is and advance to the next position
|
|
||||||
{ |
|
||||||
// but first, store read file data in cache
|
|
||||||
_cache.insert(_cache.begin(), buf + (&data[len - 1] - pTemplateStart), buf + (&data[len - 1] - pTemplateStart) + readFromCacheOrContent); |
|
||||||
++pTemplateStart; |
|
||||||
} |
|
||||||
} |
|
||||||
else // closing placeholder not found in content data, store found percent symbol as is and advance to the next position
|
|
||||||
++pTemplateStart; |
|
||||||
} |
|
||||||
else // closing placeholder not found in content data, store found percent symbol as is and advance to the next position
|
|
||||||
++pTemplateStart; |
|
||||||
if(paramName.length()) { |
|
||||||
// call callback and replace with result.
|
|
||||||
// Everything in range [pTemplateStart, pTemplateEnd] can be safely replaced with parameter value.
|
|
||||||
// Data after pTemplateEnd may need to be moved.
|
|
||||||
// The first byte of data after placeholder is located at pTemplateEnd + 1.
|
|
||||||
// It should be located at pTemplateStart + numBytesCopied (to begin right after inserted parameter value).
|
|
||||||
const String paramValue(_callback(paramName)); |
|
||||||
const char* pvstr = paramValue.c_str(); |
|
||||||
const unsigned int pvlen = paramValue.length(); |
|
||||||
const size_t numBytesCopied = std::min(pvlen, static_cast<unsigned int>(&data[originalLen - 1] - pTemplateStart + 1)); |
|
||||||
// make room for param value
|
|
||||||
// 1. move extra data to cache if parameter value is longer than placeholder AND if there is no room to store
|
|
||||||
if((pTemplateEnd + 1 < pTemplateStart + numBytesCopied) && (originalLen - (pTemplateStart + numBytesCopied - pTemplateEnd - 1) < len)) { |
|
||||||
_cache.insert(_cache.begin(), &data[originalLen - (pTemplateStart + numBytesCopied - pTemplateEnd - 1)], &data[len]); |
|
||||||
//2. parameter value is longer than placeholder text, push the data after placeholder which not saved into cache further to the end
|
|
||||||
memmove(pTemplateStart + numBytesCopied, pTemplateEnd + 1, &data[originalLen] - pTemplateStart - numBytesCopied); |
|
||||||
len = originalLen; // fix issue with truncated data, not sure if it has any side effects
|
|
||||||
} else if(pTemplateEnd + 1 != pTemplateStart + numBytesCopied) |
|
||||||
//2. Either parameter value is shorter than placeholder text OR there is enough free space in buffer to fit.
|
|
||||||
// Move the entire data after the placeholder
|
|
||||||
memmove(pTemplateStart + numBytesCopied, pTemplateEnd + 1, &data[len] - pTemplateEnd - 1); |
|
||||||
// 3. replace placeholder with actual value
|
|
||||||
memcpy(pTemplateStart, pvstr, numBytesCopied); |
|
||||||
// If result is longer than buffer, copy the remainder into cache (this could happen only if placeholder text itself did not fit entirely in buffer)
|
|
||||||
if(numBytesCopied < pvlen) { |
|
||||||
_cache.insert(_cache.begin(), pvstr + numBytesCopied, pvstr + pvlen); |
|
||||||
} else if(pTemplateStart + numBytesCopied < pTemplateEnd + 1) { // result is copied fully; if result is shorter than placeholder text...
|
|
||||||
// there is some free room, fill it from cache
|
|
||||||
const size_t roomFreed = pTemplateEnd + 1 - pTemplateStart - numBytesCopied; |
|
||||||
const size_t totalFreeRoom = originalLen - len + roomFreed; |
|
||||||
len += _readDataFromCacheOrContent(&data[len - roomFreed], totalFreeRoom) - roomFreed; |
|
||||||
} else { // result is copied fully; it is longer than placeholder text
|
|
||||||
const size_t roomTaken = pTemplateStart + numBytesCopied - pTemplateEnd - 1; |
|
||||||
len = std::min(len + roomTaken, originalLen); |
|
||||||
} |
|
||||||
} |
|
||||||
} // while(pTemplateStart)
|
|
||||||
return len; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* File Response |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncFileResponse::~AsyncFileResponse(){ |
|
||||||
if(_content) |
|
||||||
_content.close(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncFileResponse::_setContentType(const String& path){ |
|
||||||
if (path.endsWith(".html")) _contentType = "text/html"; |
|
||||||
else if (path.endsWith(".htm")) _contentType = "text/html"; |
|
||||||
else if (path.endsWith(".css")) _contentType = "text/css"; |
|
||||||
else if (path.endsWith(".json")) _contentType = "application/json"; |
|
||||||
else if (path.endsWith(".js")) _contentType = "application/javascript"; |
|
||||||
else if (path.endsWith(".png")) _contentType = "image/png"; |
|
||||||
else if (path.endsWith(".gif")) _contentType = "image/gif"; |
|
||||||
else if (path.endsWith(".jpg")) _contentType = "image/jpeg"; |
|
||||||
else if (path.endsWith(".ico")) _contentType = "image/x-icon"; |
|
||||||
else if (path.endsWith(".svg")) _contentType = "image/svg+xml"; |
|
||||||
else if (path.endsWith(".eot")) _contentType = "font/eot"; |
|
||||||
else if (path.endsWith(".woff")) _contentType = "font/woff"; |
|
||||||
else if (path.endsWith(".woff2")) _contentType = "font/woff2"; |
|
||||||
else if (path.endsWith(".ttf")) _contentType = "font/ttf"; |
|
||||||
else if (path.endsWith(".xml")) _contentType = "text/xml"; |
|
||||||
else if (path.endsWith(".pdf")) _contentType = "application/pdf"; |
|
||||||
else if (path.endsWith(".zip")) _contentType = "application/zip"; |
|
||||||
else if(path.endsWith(".gz")) _contentType = "application/x-gzip"; |
|
||||||
else _contentType = "text/plain"; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncFileResponse::AsyncFileResponse(FS &fs, const String& path, const String& contentType, bool download, AwsTemplateProcessor callback): AsyncAbstractResponse(callback){ |
|
||||||
_code = 200; |
|
||||||
_path = path; |
|
||||||
|
|
||||||
if(!download && !fs.exists(_path) && fs.exists(_path+".gz")){ |
|
||||||
_path = _path+".gz"; |
|
||||||
addHeader("Content-Encoding", "gzip"); |
|
||||||
_callback = nullptr; // Unable to process zipped templates
|
|
||||||
_sendContentLength = true; |
|
||||||
_chunked = false; |
|
||||||
} |
|
||||||
|
|
||||||
_content = fs.open(_path, "r"); |
|
||||||
_contentLength = _content.size(); |
|
||||||
|
|
||||||
if(contentType == "") |
|
||||||
_setContentType(path); |
|
||||||
else |
|
||||||
_contentType = contentType; |
|
||||||
|
|
||||||
int filenameStart = path.lastIndexOf('/') + 1; |
|
||||||
char buf[26+path.length()-filenameStart]; |
|
||||||
char* filename = (char*)path.c_str() + filenameStart; |
|
||||||
|
|
||||||
if(download) { |
|
||||||
// set filename and force download
|
|
||||||
snprintf(buf, sizeof (buf), "attachment; filename=\"%s\"", filename); |
|
||||||
} else { |
|
||||||
// set filename and force rendering
|
|
||||||
snprintf(buf, sizeof (buf), "inline; filename=\"%s\"", filename); |
|
||||||
} |
|
||||||
addHeader("Content-Disposition", buf); |
|
||||||
} |
|
||||||
|
|
||||||
AsyncFileResponse::AsyncFileResponse(File content, const String& path, const String& contentType, bool download, AwsTemplateProcessor callback): AsyncAbstractResponse(callback){ |
|
||||||
_code = 200; |
|
||||||
_path = path; |
|
||||||
|
|
||||||
if(!download && String(content.name()).endsWith(".gz") && !path.endsWith(".gz")){ |
|
||||||
addHeader("Content-Encoding", "gzip"); |
|
||||||
_callback = nullptr; // Unable to process gzipped templates
|
|
||||||
_sendContentLength = true; |
|
||||||
_chunked = false; |
|
||||||
} |
|
||||||
|
|
||||||
_content = content; |
|
||||||
_contentLength = _content.size(); |
|
||||||
|
|
||||||
if(contentType == "") |
|
||||||
_setContentType(path); |
|
||||||
else |
|
||||||
_contentType = contentType; |
|
||||||
|
|
||||||
int filenameStart = path.lastIndexOf('/') + 1; |
|
||||||
char buf[26+path.length()-filenameStart]; |
|
||||||
char* filename = (char*)path.c_str() + filenameStart; |
|
||||||
|
|
||||||
if(download) { |
|
||||||
snprintf(buf, sizeof (buf), "attachment; filename=\"%s\"", filename); |
|
||||||
} else { |
|
||||||
snprintf(buf, sizeof (buf), "inline; filename=\"%s\"", filename); |
|
||||||
} |
|
||||||
addHeader("Content-Disposition", buf); |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncFileResponse::_fillBuffer(uint8_t *data, size_t len){ |
|
||||||
return _content.read(data, len); |
|
||||||
} |
|
||||||
|
|
||||||
/*
|
|
||||||
* Stream Response |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncStreamResponse::AsyncStreamResponse(Stream &stream, const String& contentType, size_t len, AwsTemplateProcessor callback): AsyncAbstractResponse(callback) { |
|
||||||
_code = 200; |
|
||||||
_content = &stream; |
|
||||||
_contentLength = len; |
|
||||||
_contentType = contentType; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncStreamResponse::_fillBuffer(uint8_t *data, size_t len){ |
|
||||||
size_t available = _content->available(); |
|
||||||
size_t outLen = (available > len)?len:available; |
|
||||||
size_t i; |
|
||||||
for(i=0;i<outLen;i++) |
|
||||||
data[i] = _content->read(); |
|
||||||
return outLen; |
|
||||||
} |
|
||||||
|
|
||||||
/*
|
|
||||||
* Callback Response |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncCallbackResponse::AsyncCallbackResponse(const String& contentType, size_t len, AwsResponseFiller callback, AwsTemplateProcessor templateCallback): AsyncAbstractResponse(templateCallback) { |
|
||||||
_code = 200; |
|
||||||
_content = callback; |
|
||||||
_contentLength = len; |
|
||||||
if(!len) |
|
||||||
_sendContentLength = false; |
|
||||||
_contentType = contentType; |
|
||||||
_filledLength = 0; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncCallbackResponse::_fillBuffer(uint8_t *data, size_t len){ |
|
||||||
size_t ret = _content(data, len, _filledLength); |
|
||||||
if(ret != RESPONSE_TRY_AGAIN){ |
|
||||||
_filledLength += ret; |
|
||||||
} |
|
||||||
return ret; |
|
||||||
} |
|
||||||
|
|
||||||
/*
|
|
||||||
* Chunked Response |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncChunkedResponse::AsyncChunkedResponse(const String& contentType, AwsResponseFiller callback, AwsTemplateProcessor processorCallback): AsyncAbstractResponse(processorCallback) { |
|
||||||
_code = 200; |
|
||||||
_content = callback; |
|
||||||
_contentLength = 0; |
|
||||||
_contentType = contentType; |
|
||||||
_sendContentLength = false; |
|
||||||
_chunked = true; |
|
||||||
_filledLength = 0; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncChunkedResponse::_fillBuffer(uint8_t *data, size_t len){ |
|
||||||
size_t ret = _content(data, len, _filledLength); |
|
||||||
if(ret != RESPONSE_TRY_AGAIN){ |
|
||||||
_filledLength += ret; |
|
||||||
} |
|
||||||
return ret; |
|
||||||
} |
|
||||||
|
|
||||||
/*
|
|
||||||
* Progmem Response |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncProgmemResponse::AsyncProgmemResponse(int code, const String& contentType, const uint8_t * content, size_t len, AwsTemplateProcessor callback): AsyncAbstractResponse(callback) { |
|
||||||
_code = code; |
|
||||||
_content = content; |
|
||||||
_contentType = contentType; |
|
||||||
_contentLength = len; |
|
||||||
_readLength = 0; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncProgmemResponse::_fillBuffer(uint8_t *data, size_t len){ |
|
||||||
size_t left = _contentLength - _readLength; |
|
||||||
if (left > len) { |
|
||||||
memcpy_P(data, _content + _readLength, len); |
|
||||||
_readLength += len; |
|
||||||
return len; |
|
||||||
} |
|
||||||
memcpy_P(data, _content + _readLength, left); |
|
||||||
_readLength += left; |
|
||||||
return left; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Response Stream (You can print/write/printf to it, up to the contentLen bytes) |
|
||||||
* */ |
|
||||||
|
|
||||||
AsyncResponseStream::AsyncResponseStream(const String& contentType, size_t bufferSize){ |
|
||||||
_code = 200; |
|
||||||
_contentLength = 0; |
|
||||||
_contentType = contentType; |
|
||||||
_content = new cbuf(bufferSize); |
|
||||||
} |
|
||||||
|
|
||||||
AsyncResponseStream::~AsyncResponseStream(){ |
|
||||||
delete _content; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncResponseStream::_fillBuffer(uint8_t *buf, size_t maxLen){ |
|
||||||
return _content->read((char*)buf, maxLen); |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncResponseStream::write(const uint8_t *data, size_t len){ |
|
||||||
if(_started()) |
|
||||||
return 0; |
|
||||||
|
|
||||||
if(len > _content->room()){ |
|
||||||
size_t needed = len - _content->room(); |
|
||||||
_content->resizeAdd(needed); |
|
||||||
} |
|
||||||
size_t written = _content->write((const char*)data, len); |
|
||||||
_contentLength += written; |
|
||||||
return written; |
|
||||||
} |
|
||||||
|
|
||||||
size_t AsyncResponseStream::write(uint8_t data){ |
|
||||||
return write(&data, 1); |
|
||||||
} |
|
@ -1,193 +0,0 @@ |
|||||||
/*
|
|
||||||
Asynchronous WebServer library for Espressif MCUs |
|
||||||
|
|
||||||
Copyright (c) 2016 Hristo Gochkov. All rights reserved. |
|
||||||
This file is part of the esp8266 core for Arduino environment. |
|
||||||
|
|
||||||
This library is free software; you can redistribute it and/or |
|
||||||
modify it under the terms of the GNU Lesser General Public |
|
||||||
License as published by the Free Software Foundation; either |
|
||||||
version 2.1 of the License, or (at your option) any later version. |
|
||||||
|
|
||||||
This library is distributed in the hope that it will be useful, |
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
||||||
Lesser General Public License for more details. |
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public |
|
||||||
License along with this library; if not, write to the Free Software |
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|
||||||
*/ |
|
||||||
#include "ESPAsyncWebServer.h" |
|
||||||
#include "WebHandlerImpl.h" |
|
||||||
|
|
||||||
bool ON_STA_FILTER(AsyncWebServerRequest *request) { |
|
||||||
return WiFi.localIP() == request->client()->localIP(); |
|
||||||
} |
|
||||||
|
|
||||||
bool ON_AP_FILTER(AsyncWebServerRequest *request) { |
|
||||||
return WiFi.localIP() != request->client()->localIP(); |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
AsyncWebServer::AsyncWebServer(uint16_t port) |
|
||||||
: _server(port) |
|
||||||
, _rewrites(LinkedList<AsyncWebRewrite*>([](AsyncWebRewrite* r){ delete r; })) |
|
||||||
, _handlers(LinkedList<AsyncWebHandler*>([](AsyncWebHandler* h){ delete h; })) |
|
||||||
{ |
|
||||||
_catchAllHandler = new AsyncCallbackWebHandler(); |
|
||||||
if(_catchAllHandler == NULL) |
|
||||||
return; |
|
||||||
_server.onClient([](void *s, AsyncClient* c){ |
|
||||||
if(c == NULL) |
|
||||||
return; |
|
||||||
c->setRxTimeout(3); |
|
||||||
AsyncWebServerRequest *r = new AsyncWebServerRequest((AsyncWebServer*)s, c); |
|
||||||
if(r == NULL){ |
|
||||||
c->close(true); |
|
||||||
c->free(); |
|
||||||
delete c; |
|
||||||
} |
|
||||||
}, this); |
|
||||||
} |
|
||||||
|
|
||||||
AsyncWebServer::~AsyncWebServer(){ |
|
||||||
reset();
|
|
||||||
end(); |
|
||||||
if(_catchAllHandler) delete _catchAllHandler; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncWebRewrite& AsyncWebServer::addRewrite(AsyncWebRewrite* rewrite){ |
|
||||||
_rewrites.add(rewrite); |
|
||||||
return *rewrite; |
|
||||||
} |
|
||||||
|
|
||||||
bool AsyncWebServer::removeRewrite(AsyncWebRewrite *rewrite){ |
|
||||||
return _rewrites.remove(rewrite); |
|
||||||
} |
|
||||||
|
|
||||||
AsyncWebRewrite& AsyncWebServer::rewrite(const char* from, const char* to){ |
|
||||||
return addRewrite(new AsyncWebRewrite(from, to)); |
|
||||||
} |
|
||||||
|
|
||||||
AsyncWebHandler& AsyncWebServer::addHandler(AsyncWebHandler* handler){ |
|
||||||
_handlers.add(handler); |
|
||||||
return *handler; |
|
||||||
} |
|
||||||
|
|
||||||
bool AsyncWebServer::removeHandler(AsyncWebHandler *handler){ |
|
||||||
return _handlers.remove(handler); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::begin(){ |
|
||||||
_server.setNoDelay(true); |
|
||||||
_server.begin(); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::end(){ |
|
||||||
_server.end(); |
|
||||||
} |
|
||||||
|
|
||||||
#if ASYNC_TCP_SSL_ENABLED |
|
||||||
void AsyncWebServer::onSslFileRequest(AcSSlFileHandler cb, void* arg){ |
|
||||||
_server.onSslFileRequest(cb, arg); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::beginSecure(const char *cert, const char *key, const char *password){ |
|
||||||
_server.beginSecure(cert, key, password); |
|
||||||
} |
|
||||||
#endif |
|
||||||
|
|
||||||
void AsyncWebServer::_handleDisconnect(AsyncWebServerRequest *request){ |
|
||||||
delete request; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::_rewriteRequest(AsyncWebServerRequest *request){ |
|
||||||
for(const auto& r: _rewrites){ |
|
||||||
if (r->match(request)){ |
|
||||||
request->_url = r->toUrl(); |
|
||||||
request->_addGetParams(r->params()); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::_attachHandler(AsyncWebServerRequest *request){ |
|
||||||
for(const auto& h: _handlers){ |
|
||||||
if (h->filter(request) && h->canHandle(request)){ |
|
||||||
request->setHandler(h); |
|
||||||
return; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
request->addInterestingHeader("ANY"); |
|
||||||
request->setHandler(_catchAllHandler); |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
AsyncCallbackWebHandler& AsyncWebServer::on(const char* uri, WebRequestMethodComposite method, ArRequestHandlerFunction onRequest, ArUploadHandlerFunction onUpload, ArBodyHandlerFunction onBody){ |
|
||||||
AsyncCallbackWebHandler* handler = new AsyncCallbackWebHandler(); |
|
||||||
handler->setUri(uri); |
|
||||||
handler->setMethod(method); |
|
||||||
handler->onRequest(onRequest); |
|
||||||
handler->onUpload(onUpload); |
|
||||||
handler->onBody(onBody); |
|
||||||
addHandler(handler); |
|
||||||
return *handler; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncCallbackWebHandler& AsyncWebServer::on(const char* uri, WebRequestMethodComposite method, ArRequestHandlerFunction onRequest, ArUploadHandlerFunction onUpload){ |
|
||||||
AsyncCallbackWebHandler* handler = new AsyncCallbackWebHandler(); |
|
||||||
handler->setUri(uri); |
|
||||||
handler->setMethod(method); |
|
||||||
handler->onRequest(onRequest); |
|
||||||
handler->onUpload(onUpload); |
|
||||||
addHandler(handler); |
|
||||||
return *handler; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncCallbackWebHandler& AsyncWebServer::on(const char* uri, WebRequestMethodComposite method, ArRequestHandlerFunction onRequest){ |
|
||||||
AsyncCallbackWebHandler* handler = new AsyncCallbackWebHandler(); |
|
||||||
handler->setUri(uri); |
|
||||||
handler->setMethod(method); |
|
||||||
handler->onRequest(onRequest); |
|
||||||
addHandler(handler); |
|
||||||
return *handler; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncCallbackWebHandler& AsyncWebServer::on(const char* uri, ArRequestHandlerFunction onRequest){ |
|
||||||
AsyncCallbackWebHandler* handler = new AsyncCallbackWebHandler(); |
|
||||||
handler->setUri(uri); |
|
||||||
handler->onRequest(onRequest); |
|
||||||
addHandler(handler); |
|
||||||
return *handler; |
|
||||||
} |
|
||||||
|
|
||||||
AsyncStaticWebHandler& AsyncWebServer::serveStatic(const char* uri, fs::FS& fs, const char* path, const char* cache_control){ |
|
||||||
AsyncStaticWebHandler* handler = new AsyncStaticWebHandler(uri, fs, path, cache_control); |
|
||||||
addHandler(handler); |
|
||||||
return *handler; |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::onNotFound(ArRequestHandlerFunction fn){ |
|
||||||
_catchAllHandler->onRequest(fn); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::onFileUpload(ArUploadHandlerFunction fn){ |
|
||||||
_catchAllHandler->onUpload(fn); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::onRequestBody(ArBodyHandlerFunction fn){ |
|
||||||
_catchAllHandler->onBody(fn); |
|
||||||
} |
|
||||||
|
|
||||||
void AsyncWebServer::reset(){ |
|
||||||
_rewrites.free(); |
|
||||||
_handlers.free(); |
|
||||||
|
|
||||||
if (_catchAllHandler != NULL){ |
|
||||||
_catchAllHandler->onRequest(NULL); |
|
||||||
_catchAllHandler->onUpload(NULL); |
|
||||||
_catchAllHandler->onBody(NULL); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
@ -1,627 +0,0 @@ |
|||||||
<!--This is the plain html source of the hex encoded Editor-Page embedded in SPIFFSEditor.cpp --> |
|
||||||
<!DOCTYPE html> |
|
||||||
<html lang="en"> |
|
||||||
<head> |
|
||||||
<title>ESP Editor</title> |
|
||||||
<style type="text/css" media="screen"> |
|
||||||
.cm { |
|
||||||
z-index: 300; |
|
||||||
position: absolute; |
|
||||||
left: 5px; |
|
||||||
border: 1px solid #444; |
|
||||||
background-color: #F5F5F5; |
|
||||||
display: none; |
|
||||||
box-shadow: 0 0 10px rgba( 0, 0, 0, .4 ); |
|
||||||
font-size: 12px; |
|
||||||
font-family: sans-serif; |
|
||||||
font-weight:bold; |
|
||||||
} |
|
||||||
.cm ul { |
|
||||||
list-style: none; |
|
||||||
top: 0; |
|
||||||
left: 0; |
|
||||||
margin: 0; |
|
||||||
padding: 0; |
|
||||||
} |
|
||||||
.cm li { |
|
||||||
position: relative; |
|
||||||
min-width: 60px; |
|
||||||
cursor: pointer; |
|
||||||
} |
|
||||||
.cm span { |
|
||||||
color: #444; |
|
||||||
display: inline-block; |
|
||||||
padding: 6px; |
|
||||||
} |
|
||||||
.cm li:hover { background: #444; } |
|
||||||
.cm li:hover span { color: #EEE; } |
|
||||||
.tvu ul, .tvu li { |
|
||||||
padding: 0; |
|
||||||
margin: 0; |
|
||||||
list-style: none; |
|
||||||
} |
|
||||||
.tvu input { |
|
||||||
position: absolute; |
|
||||||
opacity: 0; |
|
||||||
} |
|
||||||
.tvu { |
|
||||||
font: normal 12px Verdana, Arial, Sans-serif; |
|
||||||
-moz-user-select: none; |
|
||||||
-webkit-user-select: none; |
|
||||||
user-select: none; |
|
||||||
color: #444; |
|
||||||
line-height: 16px; |
|
||||||
} |
|
||||||
.tvu span { |
|
||||||
margin-bottom:5px; |
|
||||||
padding: 0 0 0 18px; |
|
||||||
cursor: pointer; |
|
||||||
display: inline-block; |
|
||||||
height: 16px; |
|
||||||
vertical-align: middle; |
|
||||||
background: url('') no-repeat; |
|
||||||
background-position: 0px 0px; |
|
||||||
} |
|
||||||
.tvu span:hover { |
|
||||||
text-decoration: underline; |
|
||||||
} |
|
||||||
@media screen and (-webkit-min-device-pixel-ratio:0){ |
|
||||||
.tvu{ |
|
||||||
-webkit-animation: webkit-adjacent-element-selector-bugfix infinite 1s; |
|
||||||
} |
|
||||||
|
|
||||||
@-webkit-keyframes webkit-adjacent-element-selector-bugfix { |
|
||||||
from { |
|
||||||
padding: 0; |
|
||||||
} |
|
||||||
to { |
|
||||||
padding: 0; |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
#uploader { |
|
||||||
position: absolute; |
|
||||||
top: 0; |
|
||||||
right: 0; |
|
||||||
left: 0; |
|
||||||
height:28px; |
|
||||||
line-height: 24px; |
|
||||||
padding-left: 10px; |
|
||||||
background-color: #444; |
|
||||||
color:#EEE; |
|
||||||
} |
|
||||||
#tree { |
|
||||||
position: absolute; |
|
||||||
top: 28px; |
|
||||||
bottom: 0; |
|
||||||
left: 0; |
|
||||||
width:160px; |
|
||||||
padding: 8px; |
|
||||||
} |
|
||||||
#editor, #preview { |
|
||||||
position: absolute; |
|
||||||
top: 28px; |
|
||||||
right: 0; |
|
||||||
bottom: 0; |
|
||||||
left: 160px; |
|
||||||
border-left:1px solid #EEE; |
|
||||||
} |
|
||||||
#preview { |
|
||||||
background-color: #EEE; |
|
||||||
padding:5px; |
|
||||||
} |
|
||||||
#loader { |
|
||||||
position: absolute; |
|
||||||
top: 36%; |
|
||||||
right: 40%; |
|
||||||
} |
|
||||||
.loader { |
|
||||||
z-index: 10000; |
|
||||||
border: 8px solid #b5b5b5; /* Grey */ |
|
||||||
border-top: 8px solid #3498db; /* Blue */ |
|
||||||
border-bottom: 8px solid #3498db; /* Blue */ |
|
||||||
border-radius: 50%; |
|
||||||
width: 240px; |
|
||||||
height: 240px; |
|
||||||
animation: spin 2s linear infinite; |
|
||||||
display:none; |
|
||||||
} |
|
||||||
|
|
||||||
@keyframes spin { |
|
||||||
0% { transform: rotate(0deg); } |
|
||||||
100% { transform: rotate(360deg); } |
|
||||||
} |
|
||||||
</style> |
|
||||||
<script> |
|
||||||
if (typeof XMLHttpRequest === "undefined") { |
|
||||||
XMLHttpRequest = function () { |
|
||||||
try { return new ActiveXObject("Msxml2.XMLHTTP.6.0"); } catch (e) {} |
|
||||||
try { return new ActiveXObject("Msxml2.XMLHTTP.3.0"); } catch (e) {} |
|
||||||
try { return new ActiveXObject("Microsoft.XMLHTTP"); } catch (e) {} |
|
||||||
throw new Error("This browser does not support XMLHttpRequest."); |
|
||||||
}; |
|
||||||
} |
|
||||||
|
|
||||||
function ge(a){ |
|
||||||
return document.getElementById(a); |
|
||||||
} |
|
||||||
function ce(a){ |
|
||||||
return document.createElement(a); |
|
||||||
} |
|
||||||
|
|
||||||
function sortByKey(array, key) { |
|
||||||
return array.sort(function(a, b) { |
|
||||||
var x = a[key]; var y = b[key]; |
|
||||||
return ((x < y) ? -1 : ((x > y) ? 1 : 0)); |
|
||||||
}); |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
var QueuedRequester = function () { |
|
||||||
this.queue = []; |
|
||||||
this.running = false; |
|
||||||
this.xmlhttp = null; |
|
||||||
} |
|
||||||
QueuedRequester.prototype = { |
|
||||||
_request: function(req){ |
|
||||||
this.running = true; |
|
||||||
if(!req instanceof Object) return; |
|
||||||
var that = this; |
|
||||||
|
|
||||||
function ajaxCb(x,d){ return function(){ |
|
||||||
if (x.readyState == 4){ |
|
||||||
ge("loader").style.display = "none"; |
|
||||||
d.callback(x.status, x.responseText); |
|
||||||
if(that.queue.length === 0) that.running = false; |
|
||||||
if(that.running) that._request(that.queue.shift()); |
|
||||||
} |
|
||||||
}} |
|
||||||
|
|
||||||
ge("loader").style.display = "block"; |
|
||||||
|
|
||||||
var p = ""; |
|
||||||
if(req.params instanceof FormData){ |
|
||||||
p = req.params; |
|
||||||
} else if(req.params instanceof Object){ |
|
||||||
for (var key in req.params) { |
|
||||||
if(p === "") |
|
||||||
p += (req.method === "GET")?"?":""; |
|
||||||
else |
|
||||||
p += "&"; |
|
||||||
p += encodeURIComponent(key)+"="+encodeURIComponent(req.params[key]); |
|
||||||
}; |
|
||||||
} |
|
||||||
|
|
||||||
this.xmlhttp = new XMLHttpRequest(); |
|
||||||
this.xmlhttp.onreadystatechange = ajaxCb(this.xmlhttp, req); |
|
||||||
if(req.method === "GET"){ |
|
||||||
this.xmlhttp.open(req.method, req.url+p, true); |
|
||||||
this.xmlhttp.send(); |
|
||||||
} else { |
|
||||||
this.xmlhttp.open(req.method, req.url, true); |
|
||||||
if(p instanceof String) |
|
||||||
this.xmlhttp.setRequestHeader("Content-type", "application/x-www-form-urlencoded"); |
|
||||||
this.xmlhttp.send(p); |
|
||||||
} |
|
||||||
}, |
|
||||||
stop: function(){ |
|
||||||
if(this.running) this.running = false; |
|
||||||
if(this.xmlhttp && this.xmlhttp.readyState < 4){ |
|
||||||
this.xmlhttp.abort(); |
|
||||||
} |
|
||||||
}, |
|
||||||
add: function(method, url, params, callback){ |
|
||||||
this.queue.push({url:url,method:method,params:params,callback:callback}); |
|
||||||
if(!this.running){ |
|
||||||
this._request(this.queue.shift()); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
var requests = new QueuedRequester(); |
|
||||||
|
|
||||||
function createFileUploader(element, tree, editor){ |
|
||||||
var xmlHttp; |
|
||||||
|
|
||||||
var refresh = ce("button"); |
|
||||||
refresh.innerHTML = 'Refresh List'; |
|
||||||
ge(element).appendChild(refresh); |
|
||||||
|
|
||||||
var input = ce("input"); |
|
||||||
input.type = "file"; |
|
||||||
input.multiple = false; |
|
||||||
input.name = "data"; |
|
||||||
input.id="upload-select"; |
|
||||||
ge(element).appendChild(input); |
|
||||||
|
|
||||||
var path = ce("input"); |
|
||||||
path.id = "upload-path"; |
|
||||||
path.type = "text"; |
|
||||||
path.name = "path"; |
|
||||||
path.defaultValue = "/"; |
|
||||||
ge(element).appendChild(path); |
|
||||||
|
|
||||||
var button = ce("button"); |
|
||||||
button.innerHTML = 'Upload'; |
|
||||||
ge(element).appendChild(button); |
|
||||||
|
|
||||||
var mkfile = ce("button"); |
|
||||||
mkfile.innerHTML = 'Create'; |
|
||||||
ge(element).appendChild(mkfile); |
|
||||||
|
|
||||||
var filename = ce("input"); |
|
||||||
filename.id = "editor-filename"; |
|
||||||
filename.type = "text"; |
|
||||||
filename.disabled= true; |
|
||||||
filename.size = 20; |
|
||||||
ge(element).appendChild(filename); |
|
||||||
|
|
||||||
var savefile = ce("button"); |
|
||||||
savefile.innerHTML = ' Save ' ; |
|
||||||
ge(element).appendChild(savefile); |
|
||||||
|
|
||||||
function httpPostProcessRequest(status, responseText){ |
|
||||||
if(status != 200) |
|
||||||
alert("ERROR["+status+"]: "+responseText); |
|
||||||
else |
|
||||||
tree.refreshPath(path.value); |
|
||||||
} |
|
||||||
function createPath(p){ |
|
||||||
var formData = new FormData(); |
|
||||||
formData.append("path", p); |
|
||||||
requests.add("PUT", "/edit", formData, httpPostProcessRequest); |
|
||||||
} |
|
||||||
|
|
||||||
mkfile.onclick = function(e){ |
|
||||||
createPath(path.value); |
|
||||||
editor.loadUrl(path.value); |
|
||||||
path.value="/"; |
|
||||||
}; |
|
||||||
|
|
||||||
savefile.onclick = function(e){ |
|
||||||
editor.execCommand('saveCommand'); |
|
||||||
}; |
|
||||||
|
|
||||||
refresh.onclick = function(e){ |
|
||||||
tree.refreshPath(path.value); |
|
||||||
}; |
|
||||||
|
|
||||||
button.onclick = function(e){ |
|
||||||
if(input.files.length === 0){ |
|
||||||
return; |
|
||||||
} |
|
||||||
var formData = new FormData(); |
|
||||||
formData.append("data", input.files[0], path.value); |
|
||||||
requests.add("POST", "/edit", formData, httpPostProcessRequest); |
|
||||||
var uploadPath= ge("upload-path"); |
|
||||||
uploadPath.value="/"; |
|
||||||
var uploadSelect= ge("upload-select"); |
|
||||||
uploadSelect.value=""; |
|
||||||
}; |
|
||||||
input.onchange = function(e){ |
|
||||||
if(input.files.length === 0) return; |
|
||||||
var filename = input.files[0].name; |
|
||||||
var ext = /(?:\.([^.]+))?$/.exec(filename)[1]; |
|
||||||
var name = /(.*)\.[^.]+$/.exec(filename)[1]; |
|
||||||
if(typeof name !== undefined){ |
|
||||||
filename = name; |
|
||||||
} |
|
||||||
path.value = "/"+filename+"."+ext; |
|
||||||
}; |
|
||||||
} |
|
||||||
|
|
||||||
function createTree(element, editor){ |
|
||||||
var preview = ge("preview"); |
|
||||||
var treeRoot = ce("div"); |
|
||||||
treeRoot.className = "tvu"; |
|
||||||
ge(element).appendChild(treeRoot); |
|
||||||
|
|
||||||
function loadDownload(path){ |
|
||||||
ge('download-frame').src = "/edit?download="+path; |
|
||||||
} |
|
||||||
|
|
||||||
function loadPreview(path){ |
|
||||||
var edfname = ge("editor-filename"); |
|
||||||
edfname.value=path; |
|
||||||
ge("editor").style.display = "none"; |
|
||||||
preview.style.display = "block"; |
|
||||||
preview.innerHTML = '<img src="/edit?edit='+path+'&_cb='+Date.now()+'" style="max-width:100%; max-height:100%; margin:auto; display:block;" />'; |
|
||||||
} |
|
||||||
|
|
||||||
function fillFileMenu(el, path){ |
|
||||||
var list = ce("ul"); |
|
||||||
el.appendChild(list); |
|
||||||
var action = ce("li"); |
|
||||||
list.appendChild(action); |
|
||||||
if(isImageFile(path)){ |
|
||||||
action.innerHTML = "<span>Preview</span>"; |
|
||||||
action.onclick = function(e){ |
|
||||||
loadPreview(path); |
|
||||||
if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el); |
|
||||||
}; |
|
||||||
} else if(isTextFile(path)){ |
|
||||||
action.innerHTML = "<span>Edit</span>"; |
|
||||||
action.onclick = function(e){ |
|
||||||
editor.loadUrl(path); |
|
||||||
if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el); |
|
||||||
}; |
|
||||||
} |
|
||||||
var download = ce("li"); |
|
||||||
list.appendChild(download); |
|
||||||
download.innerHTML = "<span>Download</span>"; |
|
||||||
download.onclick = function(e){ |
|
||||||
loadDownload(path); |
|
||||||
if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el); |
|
||||||
}; |
|
||||||
var delFile = ce("li"); |
|
||||||
list.appendChild(delFile); |
|
||||||
delFile.innerHTML = "<span>Delete</span>"; |
|
||||||
delFile.onclick = function(e){ |
|
||||||
httpDelete(path); |
|
||||||
if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(el); |
|
||||||
}; |
|
||||||
} |
|
||||||
|
|
||||||
function showContextMenu(event, path, isfile){ |
|
||||||
var divContext = ce("div"); |
|
||||||
var scrollTop = document.body.scrollTop ? document.body.scrollTop : document.documentElement.scrollTop; |
|
||||||
var scrollLeft = document.body.scrollLeft ? document.body.scrollLeft : document.documentElement.scrollLeft; |
|
||||||
var left = event.clientX + scrollLeft; |
|
||||||
var top = event.clientY + scrollTop; |
|
||||||
divContext.className = 'cm'; |
|
||||||
divContext.style.display = 'block'; |
|
||||||
divContext.style.left = left + 'px'; |
|
||||||
divContext.style.top = top + 'px'; |
|
||||||
fillFileMenu(divContext, path); |
|
||||||
document.body.appendChild(divContext); |
|
||||||
var width = divContext.offsetWidth; |
|
||||||
var height = divContext.offsetHeight; |
|
||||||
divContext.onmouseout = function(e){ |
|
||||||
if(e.clientX < left || e.clientX > (left + width) || e.clientY < top || e.clientY > (top + height)){ |
|
||||||
if(document.body.getElementsByClassName('cm').length > 0) document.body.removeChild(divContext); |
|
||||||
} |
|
||||||
}; |
|
||||||
} |
|
||||||
|
|
||||||
function createTreeLeaf(path, name, size){ |
|
||||||
var leaf = ce("li"); |
|
||||||
leaf.id = name; |
|
||||||
var label = ce("span"); |
|
||||||
label.innerHTML = name; |
|
||||||
leaf.appendChild(label); |
|
||||||
leaf.onclick = function(e){ |
|
||||||
if(isTextFile(leaf.id.toLowerCase())){ |
|
||||||
editor.loadUrl(leaf.id); |
|
||||||
} else if(isImageFile(leaf.id.toLowerCase())){ |
|
||||||
loadPreview(leaf.id); |
|
||||||
} |
|
||||||
}; |
|
||||||
leaf.oncontextmenu = function(e){ |
|
||||||
e.preventDefault(); |
|
||||||
e.stopPropagation(); |
|
||||||
showContextMenu(e, leaf.id, true); |
|
||||||
}; |
|
||||||
return leaf; |
|
||||||
} |
|
||||||
|
|
||||||
function addList(parent, path, items){ |
|
||||||
sortByKey(items, 'name'); |
|
||||||
var list = ce("ul"); |
|
||||||
parent.appendChild(list); |
|
||||||
var ll = items.length; |
|
||||||
for(var i = 0; i < ll; i++){ |
|
||||||
if(items[i].type === "file") |
|
||||||
list.appendChild(createTreeLeaf(path, items[i].name, items[i].size)); |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
|
|
||||||
function isTextFile(path){ |
|
||||||
var ext = /(?:\.([^.]+))?$/.exec(path)[1]; |
|
||||||
if(typeof ext !== undefined){ |
|
||||||
switch(ext){ |
|
||||||
case "txt": |
|
||||||
case "htm": |
|
||||||
case "html": |
|
||||||
case "js": |
|
||||||
case "css": |
|
||||||
case "xml": |
|
||||||
case "json": |
|
||||||
case "conf": |
|
||||||
case "ini": |
|
||||||
case "h": |
|
||||||
case "c": |
|
||||||
case "cpp": |
|
||||||
case "php": |
|
||||||
case "hex": |
|
||||||
case "ino": |
|
||||||
case "pde": |
|
||||||
return true; |
|
||||||
} |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
function isImageFile(path){ |
|
||||||
var ext = /(?:\.([^.]+))?$/.exec(path)[1]; |
|
||||||
if(typeof ext !== undefined){ |
|
||||||
switch(ext){ |
|
||||||
case "png": |
|
||||||
case "jpg": |
|
||||||
case "gif": |
|
||||||
case "bmp": |
|
||||||
return true; |
|
||||||
} |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
this.refreshPath = function(path){ |
|
||||||
treeRoot.removeChild(treeRoot.childNodes[0]); |
|
||||||
httpGet(treeRoot, "/"); |
|
||||||
}; |
|
||||||
|
|
||||||
function delCb(path){ |
|
||||||
return function(status, responseText){ |
|
||||||
if(status != 200){ |
|
||||||
alert("ERROR["+status+"]: "+responseText); |
|
||||||
} else { |
|
||||||
treeRoot.removeChild(treeRoot.childNodes[0]); |
|
||||||
httpGet(treeRoot, "/"); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
function httpDelete(filename){ |
|
||||||
var formData = new FormData(); |
|
||||||
formData.append("path", filename); |
|
||||||
requests.add("DELETE", "/edit", formData, delCb(filename)); |
|
||||||
} |
|
||||||
|
|
||||||
function getCb(parent, path){ |
|
||||||
return function(status, responseText){ |
|
||||||
if(status == 200) |
|
||||||
addList(parent, path, JSON.parse(responseText)); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
function httpGet(parent, path){ |
|
||||||
requests.add("GET", "/edit", { list: path }, getCb(parent, path)); |
|
||||||
} |
|
||||||
|
|
||||||
httpGet(treeRoot, "/"); |
|
||||||
return this; |
|
||||||
} |
|
||||||
|
|
||||||
function createEditor(element, file, lang, theme, type){ |
|
||||||
function getLangFromFilename(filename){ |
|
||||||
var lang = "plain"; |
|
||||||
var ext = /(?:\.([^.]+))?$/.exec(filename)[1]; |
|
||||||
if(typeof ext !== undefined){ |
|
||||||
switch(ext){ |
|
||||||
case "txt": lang = "plain"; break; |
|
||||||
case "hex": lang = "plain"; break; |
|
||||||
case "conf": lang = "plain"; break; |
|
||||||
case "htm": lang = "html"; break; |
|
||||||
case "js": lang = "javascript"; break; |
|
||||||
case "h": lang = "c_cpp"; break; |
|
||||||
case "c": lang = "c_cpp"; break; |
|
||||||
case "cpp": lang = "c_cpp"; break; |
|
||||||
case "css": |
|
||||||
case "scss": |
|
||||||
case "php": |
|
||||||
case "html": |
|
||||||
case "json": |
|
||||||
case "xml": |
|
||||||
case "ini": lang = ext; |
|
||||||
} |
|
||||||
} |
|
||||||
return lang; |
|
||||||
} |
|
||||||
|
|
||||||
if(typeof file === "undefined") file = "/index.html"; |
|
||||||
|
|
||||||
if(typeof lang === "undefined"){ |
|
||||||
lang = getLangFromFilename(file); |
|
||||||
} |
|
||||||
|
|
||||||
if(typeof theme === "undefined") theme = "textmate"; |
|
||||||
|
|
||||||
if(typeof type === "undefined"){ |
|
||||||
type = "text/"+lang; |
|
||||||
if(lang === "c_cpp") type = "text/plain"; |
|
||||||
} |
|
||||||
|
|
||||||
var editor = ace.edit(element); |
|
||||||
function httpPostProcessRequest(status, responseText){ |
|
||||||
if(status != 200) alert("ERROR["+status+"]: "+responseText); |
|
||||||
} |
|
||||||
function httpPost(filename, data, type){ |
|
||||||
var formData = new FormData(); |
|
||||||
formData.append("data", new Blob([data], { type: type }), filename); |
|
||||||
requests.add("POST", "/edit", formData, httpPostProcessRequest); |
|
||||||
} |
|
||||||
function httpGetProcessRequest(status, responseText){ |
|
||||||
ge("preview").style.display = "none"; |
|
||||||
ge("editor").style.display = "block"; |
|
||||||
if(status == 200) |
|
||||||
editor.setValue(responseText); |
|
||||||
else |
|
||||||
editor.setValue(""); |
|
||||||
editor.clearSelection(); |
|
||||||
} |
|
||||||
function httpGet(theUrl){ |
|
||||||
requests.add("GET", "/edit", { edit: theUrl }, httpGetProcessRequest); |
|
||||||
} |
|
||||||
|
|
||||||
if(lang !== "plain") editor.getSession().setMode("ace/mode/"+lang); |
|
||||||
editor.setTheme("ace/theme/"+theme); |
|
||||||
editor.$blockScrolling = Infinity; |
|
||||||
editor.getSession().setUseSoftTabs(true); |
|
||||||
editor.getSession().setTabSize(2); |
|
||||||
editor.setHighlightActiveLine(true); |
|
||||||
editor.setShowPrintMargin(false); |
|
||||||
editor.commands.addCommand({ |
|
||||||
name: 'saveCommand', |
|
||||||
bindKey: {win: 'Ctrl-S', mac: 'Command-S'}, |
|
||||||
exec: function(editor) { |
|
||||||
httpPost(file, editor.getValue()+"", type); |
|
||||||
}, |
|
||||||
readOnly: false |
|
||||||
}); |
|
||||||
editor.commands.addCommand({ |
|
||||||
name: 'undoCommand', |
|
||||||
bindKey: {win: 'Ctrl-Z', mac: 'Command-Z'}, |
|
||||||
exec: function(editor) { |
|
||||||
editor.getSession().getUndoManager().undo(false); |
|
||||||
}, |
|
||||||
readOnly: false |
|
||||||
}); |
|
||||||
editor.commands.addCommand({ |
|
||||||
name: 'redoCommand', |
|
||||||
bindKey: {win: 'Ctrl-Shift-Z', mac: 'Command-Shift-Z'}, |
|
||||||
exec: function(editor) { |
|
||||||
editor.getSession().getUndoManager().redo(false); |
|
||||||
}, |
|
||||||
readOnly: false |
|
||||||
}); |
|
||||||
editor.loadUrl = function(filename){ |
|
||||||
var edfname = ge("editor-filename"); |
|
||||||
edfname.value=filename; |
|
||||||
file = filename; |
|
||||||
lang = getLangFromFilename(file); |
|
||||||
type = "text/"+lang; |
|
||||||
if(lang !== "plain") editor.getSession().setMode("ace/mode/"+lang); |
|
||||||
httpGet(file); |
|
||||||
}; |
|
||||||
return editor; |
|
||||||
} |
|
||||||
function onBodyLoad(){ |
|
||||||
var vars = {}; |
|
||||||
var parts = window.location.href.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m,key,value) { vars[key] = value; }); |
|
||||||
var editor = createEditor("editor", vars.file, vars.lang, vars.theme); |
|
||||||
var tree = createTree("tree", editor); |
|
||||||
createFileUploader("uploader", tree, editor); |
|
||||||
if(typeof vars.file === "undefined") vars.file = "/index.htm"; |
|
||||||
editor.loadUrl(vars.file); |
|
||||||
}; |
|
||||||
</script> |
|
||||||
<script id='ace' src="https://cdnjs.cloudflare.com/ajax/libs/ace/1.2.6/ace.js" type="text/javascript" charset="utf-8"></script> |
|
||||||
<script> |
|
||||||
if (typeof ace.edit == "undefined") { |
|
||||||
var script = document.createElement('script'); |
|
||||||
script.src = "/ace.js"; |
|
||||||
script.async = false; |
|
||||||
document.head.appendChild(script); |
|
||||||
} |
|
||||||
</script> |
|
||||||
</head> |
|
||||||
<body onload="onBodyLoad();"> |
|
||||||
<div id="loader" class="loader"></div> |
|
||||||
<div id="uploader"></div> |
|
||||||
<div id="tree"></div> |
|
||||||
<div id="editor"></div> |
|
||||||
<div id="preview" style="display:none;"></div> |
|
||||||
<iframe id=download-frame style='display:none;'></iframe> |
|
||||||
</body> |
|
||||||
</html> |
|
@ -0,0 +1,25 @@ |
|||||||
|
#pragma once |
||||||
|
|
||||||
|
#include <cstdint> |
||||||
|
|
||||||
|
struct Status |
||||||
|
{ |
||||||
|
char realtime[32] = {0}; // UTC date and time, in format YYYY-MM-DDTHH:mm:ss.sssZ
|
||||||
|
|
||||||
|
uint16_t batteryVoltage = 0; // in mV
|
||||||
|
uint16_t batteryOutputCurrent = 0; // in mV
|
||||||
|
int16_t temperature = 0; // in tenth of °C
|
||||||
|
int32_t altitude = 0; // in mm above sea level (can be negative if below sea level, or depending on atmospheric conditions)
|
||||||
|
float latitude = -1000.0f; // in decimal degrees
|
||||||
|
float longitude = -1000.0f; // in decimal degrees
|
||||||
|
float gpsAltitude = -1000.0f; // in meters, above sea level
|
||||||
|
|
||||||
|
float speed = 0.0f; |
||||||
|
|
||||||
|
// current trip
|
||||||
|
uint32_t tripDistance = 0; // in meters
|
||||||
|
uint16_t tripMovingTime = 0; // cumulated seconds, only when moving at non-zero speed
|
||||||
|
uint16_t tripTotalTime = 0; // total trip time in seconds
|
||||||
|
uint32_t tripAscendingElevation = 0; // cumulated ascending elevation, in millimeters
|
||||||
|
uint32_t tripMotorEnergy = 0; // in Joules
|
||||||
|
}; |
@ -0,0 +1,175 @@ |
|||||||
|
#include "WebServer.h" |
||||||
|
|
||||||
|
#include "DebugLog.h" |
||||||
|
#include "DataLogger.h" |
||||||
|
|
||||||
|
#include <Arduino.h> |
||||||
|
#include <SPIFFS.h> |
||||||
|
#include <string> |
||||||
|
|
||||||
|
#include "HTTPServer.hpp" |
||||||
|
#include "SSLCert.hpp" |
||||||
|
#include "HTTPRequest.hpp" |
||||||
|
#include "HTTPResponse.hpp" |
||||||
|
|
||||||
|
/** Check if we have multiple cores */ |
||||||
|
#if CONFIG_FREERTOS_UNICORE |
||||||
|
#define WEBSERVER_RUNNING_CORE 0 |
||||||
|
#else |
||||||
|
#define WEBSERVER_RUNNING_CORE 1 |
||||||
|
#endif |
||||||
|
|
||||||
|
using namespace httpsserver; |
||||||
|
|
||||||
|
detail::WebServer WebServer; |
||||||
|
|
||||||
|
namespace detail |
||||||
|
{ |
||||||
|
HTTPServer httpServer; |
||||||
|
|
||||||
|
WebServer::WebServer() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
WebServer::~WebServer() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::begin() |
||||||
|
{ |
||||||
|
::DebugLog.println("Creating server task... "); |
||||||
|
xTaskCreatePinnedToCore(WebServer::ServerTask_, "webserver", 6144, nullptr, 1, nullptr, WEBSERVER_RUNNING_CORE); |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::setStatus(const Status& status) |
||||||
|
{ |
||||||
|
status_ = status; |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::ServerTask_(void* params) |
||||||
|
{ |
||||||
|
// Main HTML page
|
||||||
|
ResourceNode* nodeRoot = new ResourceNode("/", "GET", &WebServer::HandleIndex_); |
||||||
|
ResourceNode* nodeIndex = new ResourceNode("/index.html", "GET", &WebServer::HandleIndex_); |
||||||
|
httpServer.registerNode(nodeRoot); |
||||||
|
httpServer.registerNode(nodeIndex); |
||||||
|
|
||||||
|
// API
|
||||||
|
ResourceNode* nodeGetStatus = new ResourceNode("/api/status", "GET", &WebServer::HandleGetStatus_); |
||||||
|
ResourceNode* nodePostInfo = new ResourceNode("/api/info", "POST", &WebServer::HandlePostInfo_); |
||||||
|
httpServer.registerNode(nodeGetStatus); |
||||||
|
httpServer.registerNode(nodePostInfo); |
||||||
|
|
||||||
|
// Default node (either a static file, if found, or a 404 error)
|
||||||
|
ResourceNode* nodeDefault = new ResourceNode("", "GET", &WebServer::HandleDefault_); |
||||||
|
httpServer.setDefaultNode(nodeDefault); |
||||||
|
|
||||||
|
::DebugLog.println("Starting server... "); |
||||||
|
httpServer.start(); |
||||||
|
if (httpServer.isRunning()) { |
||||||
|
::DebugLog.println("Server ready."); |
||||||
|
|
||||||
|
while(true) { |
||||||
|
httpServer.loop(); |
||||||
|
delay(1); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::HandleIndex_(HTTPRequest * request, HTTPResponse * response) |
||||||
|
{ |
||||||
|
// Status code is 200 OK by default.
|
||||||
|
response->setHeader("Content-Type", "text/html"); |
||||||
|
|
||||||
|
File file = SPIFFS.open("/www/index.html"); |
||||||
|
SendContent_(file, response); |
||||||
|
file.close(); |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::HandleGetStatus_(httpsserver::HTTPRequest * request, httpsserver::HTTPResponse * response) |
||||||
|
{ |
||||||
|
const Status& status = ::WebServer.status_; |
||||||
|
|
||||||
|
int v = status.batteryVoltage; |
||||||
|
int c = status.batteryOutputCurrent; |
||||||
|
int s = (int)(status.speed * 1000.0f + 0.5f); |
||||||
|
int temp = status.temperature; |
||||||
|
int alt = status.altitude; |
||||||
|
|
||||||
|
int td = status.tripDistance; |
||||||
|
int ttt = status.tripTotalTime; |
||||||
|
int tmt = status.tripMovingTime; |
||||||
|
int tae = status.tripAscendingElevation / 100; // convert mm to dm
|
||||||
|
int tme = status.tripMotorEnergy / 360; // convert Joules to dWh (tenth of Wh)
|
||||||
|
|
||||||
|
float latitude = -1000.0f; |
||||||
|
float longitude = -1000.0f; |
||||||
|
char realtime[64] = {0}; |
||||||
|
|
||||||
|
const char* logFileName = DataLogger::get().currentLogFileName(); |
||||||
|
if(String(logFileName).startsWith("/log/")) logFileName += 5; |
||||||
|
|
||||||
|
int totalSize = (int)(SPIFFS.totalBytes() / 1000); |
||||||
|
int usedSize = (int)(SPIFFS.usedBytes() / 1000); |
||||||
|
|
||||||
|
char json[256]; |
||||||
|
sprintf(json, "{\"v\":%d,\"c\":%d,\"s\":%d,\"td\":%d,\"ttt\":%d,\"tmt\":%d,\"tae\":%d,\"tme\":%d,\"temp\":%d,\"alt\":%d,\"log\":\"%s\",\"tot\":%d,\"used\":%d,\"lat\":%.5f,\"lng\":%.5f,\"d\":\"%s\"}", v, c, s, td, ttt, tmt, tae, tme, temp, alt, logFileName, totalSize, usedSize, latitude, longitude, realtime); |
||||||
|
|
||||||
|
response->setHeader("Content-Type", "text/json"); |
||||||
|
response->print(json); |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::HandlePostInfo_(httpsserver::HTTPRequest * request, httpsserver::HTTPResponse * response) |
||||||
|
{ |
||||||
|
request->discardRequestBody(); |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::HandleDefault_(HTTPRequest * request, HTTPResponse * response) |
||||||
|
{ |
||||||
|
// Discard request body, if we received any
|
||||||
|
// We do this, as this is the default node and may also server POST/PUT requests
|
||||||
|
request->discardRequestBody(); |
||||||
|
|
||||||
|
std::string filePath = "/www"; |
||||||
|
std::string fullURL = request->getRequestString(); |
||||||
|
filePath = filePath + fullURL.substr(0, fullURL.find("?")); |
||||||
|
//::DebugLog.println(filePath.c_str());
|
||||||
|
File file = SPIFFS.open(filePath.c_str()); |
||||||
|
|
||||||
|
if(!file || file.isDirectory()) |
||||||
|
{ |
||||||
|
// We've not found any matching file, so this is a 404 error
|
||||||
|
|
||||||
|
response->setStatusCode(404); |
||||||
|
response->setStatusText("Not Found"); |
||||||
|
|
||||||
|
response->setHeader("Content-Type", "text/html"); |
||||||
|
|
||||||
|
response->println("<!DOCTYPE html>"); |
||||||
|
response->println("<html>"); |
||||||
|
response->println("<head><title>Not Found</title></head>"); |
||||||
|
response->println("<body><h1>404 Not Found</h1><p>The requested resource was not found on this server.</p></body>"); |
||||||
|
response->println("</html>"); |
||||||
|
} |
||||||
|
else |
||||||
|
{ |
||||||
|
// We've found a file, send the content
|
||||||
|
|
||||||
|
response->setHeader("Cache-Control", "max-age=5184000"); |
||||||
|
|
||||||
|
SendContent_(file, response); |
||||||
|
} |
||||||
|
|
||||||
|
file.close(); |
||||||
|
} |
||||||
|
|
||||||
|
void WebServer::SendContent_(Stream& stream, httpsserver::HTTPResponse * response) |
||||||
|
{ |
||||||
|
uint8_t buffer[512]; |
||||||
|
while(stream.available()) |
||||||
|
{ |
||||||
|
size_t numBytes = stream.readBytes(buffer, sizeof(buffer)); |
||||||
|
response->write(buffer, numBytes); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,39 @@ |
|||||||
|
#pragma once |
||||||
|
|
||||||
|
#include "Status.h" |
||||||
|
|
||||||
|
class Stream; |
||||||
|
|
||||||
|
namespace httpsserver |
||||||
|
{ |
||||||
|
class HTTPRequest; |
||||||
|
class HTTPResponse; |
||||||
|
} |
||||||
|
|
||||||
|
namespace detail |
||||||
|
{ |
||||||
|
class WebServer |
||||||
|
{ |
||||||
|
public: |
||||||
|
WebServer(); |
||||||
|
~WebServer(); |
||||||
|
|
||||||
|
void begin(); |
||||||
|
|
||||||
|
void setStatus(const Status& status); |
||||||
|
|
||||||
|
private: |
||||||
|
static void ServerTask_(void* params); |
||||||
|
static void HandleIndex_(httpsserver::HTTPRequest * request, httpsserver::HTTPResponse * response); |
||||||
|
static void HandleGetStatus_(httpsserver::HTTPRequest * request, httpsserver::HTTPResponse * response); |
||||||
|
static void HandlePostInfo_(httpsserver::HTTPRequest * request, httpsserver::HTTPResponse * response); |
||||||
|
static void HandleDefault_(httpsserver::HTTPRequest * request, httpsserver::HTTPResponse * response); |
||||||
|
|
||||||
|
static void SendContent_(Stream& stream, httpsserver::HTTPResponse * response); |
||||||
|
|
||||||
|
private: |
||||||
|
Status status_; |
||||||
|
}; |
||||||
|
} |
||||||
|
|
||||||
|
extern detail::WebServer WebServer; |
Loading…
Reference in new issue