diff --git a/.github/workflows/python_tests_32.yml b/.github/workflows/python_tests_32.yml new file mode 100644 index 00000000..67534285 --- /dev/null +++ b/.github/workflows/python_tests_32.yml @@ -0,0 +1,80 @@ +name: Python Tests 32-bit + +on: + push: + paths-ignore: + - 'docs/**' + - '.github/workflows/**' + - '!.github/workflows/python_tests_32.yml' + pull_request: + +jobs: + trick_32bit: + runs-on: ubuntu-18.04 + container: docker://centos:7 + steps: + - name: Checkout repository + uses: actions/checkout@master + - name: Add yum repo + run: yum -y install epel-release.noarch && yum -y update + - name: Install python dependencies + run: yum install -y python3-venv python3 python3-pip python3-devel gcc + - name: Info + run: | + python3 --version + - name: Build Python environment + run: | + cd share/trick/pymods/trick + python3 -m venv .venv && . .venv/bin/activate && pip3 install -r requirements.txt + - name: Install dependency group + run: yum -y groupinstall "Development tools" && yum -y update + - name: Install other dependencies + run: > + yum remove -y swig && yum install -y llvm llvm-devel llvm-static clang clang-devel + bison flex gcc gcc-c++ libxml2-devel make cmake wget + ncurses-devel openmotif openmotif-devel python-devel perl + perl-Digest-MD5 swig3 zlib-devel glibc.x86_64 libxml2-devel.i686 + ncurses-devel.i686 zlib-devel.i686 python-libs.i686 + expat-2.1.0-10.el7_3.i686 glibc-devel-2.17-196.el7.i686 + glibc.i686 glibc-devel.i686 udunits2 udunits2-devel gtest-devel.i686 + java-11-openjdk java-11-openjdk-devel expat-devel.i686 + which gcc-gfortran git wget gsl-devel gtest-devel gsl-devel.i686 + maven udunits2 udunits2-devel zip python3-tkinter xorg-x11-server-Xvfb + - name: Symlink python + run: | + cd /usr/lib + ln -s ./libpython2.7.so.1.0 libpython2.7.so + - name: Install Udunits (32 bit) + run: | + cd / + curl --retry 4 -O https://artifacts.unidata.ucar.edu/repository/downloads-udunits/udunits-2.2.28.tar.gz + tar xfvz udunits-2.2.28.tar.gz + rm -rf udunits-2.2.28.tar.gz + cd udunits-2.2.28 + export CFLAGS="-m32" + ./configure + make + make install + - name: Install GTest (32 bit) + run: | + wget https://github.com/google/googletest/archive/release-1.8.0.tar.gz + tar xfvz release-1.8.0.tar.gz + cd googletest-release-1.8.0/googletest + export CFLAGS="-m32" + export CXXFLAGS="-m32" + cmake . + make + make install + - name: Build Trick + run: | + yum -y update && yum clean all + export MAKEFLAGS=-j`nproc` + ./configure --without-hdf5 --enable-32bit + make + - name: Run civet tests + run: | + cd share/trick/pymods/trick/ + . .venv/bin/activate + ./run_tests.py + env: + TRICK_HOME: "${{ github.workspace }}" diff --git a/.github/workflows/python_tests_linux.yml b/.github/workflows/python_tests_linux.yml new file mode 100644 index 00000000..053f319f --- /dev/null +++ b/.github/workflows/python_tests_linux.yml @@ -0,0 +1,186 @@ +name: Python Tests Linux + +on: + push: + paths-ignore: + - 'docs/**' + - '.github/workflows/**' + - '!.github/workflows/python_tests_linux.yml' + pull_request: + +jobs: + build: + strategy: + fail-fast: false + matrix: + cfg: +#-------- Operating Systems ---------------- + - { os: ubuntu, tag: 18.04, arch: debian } # EOL April 2023 + - { os: ubuntu, tag: 20.04, arch: debian } # EOL April 2025 + - { os: debian, tag: 10, arch: debian } # EOL 2024 + - { os: centos, tag: 7, arch: rhel } # EOL June 2024 + - { os: centos, tag: latest, arch: rhel } # 8 as of April 2020 + # - { os: fedora, tag: latest, arch: rhel } # 31 as of April 2020 + # - { os: fedora, tag: 33, arch: rhel } # feeling confident? + # - { os: fedora, tag: rawhide, arch: rhel } # for thrill-seekers only + +#-------- Defaults -------------------------- + include: + - cfg: {} + deps: >- + bison + clang + flex + git + llvm + make + maven + cmake + zip + install_gtest: echo gtest already installed + conf_pkg: echo package manager already configured + install_cmd: install -y +#-------- Debian-based Dependencies ---------------- + - cfg: { arch: debian } + pkg_mgr: apt-get + conf_pkg: DEBIAN_FRONTEND=noninteractive && apt-get update && apt-get install -y tzdata + arch_deps: >- + swig + curl + g++ + libx11-dev + libxml2-dev + libxt-dev + libmotif-common + libmotif-dev + python2.7-dev + zlib1g-dev + llvm-dev + libclang-dev + libudunits2-dev + libgtest-dev + python3 + python3-tk + python3-venv + python3-dev + xvfb + install_gtest: cd /usr/src/gtest && cmake . && make && cp libgtest* /usr/lib/ +#-------- RHEL Dependencies ---------------- + - cfg: { arch: rhel } + arch_deps: >- + clang-devel + gcc + gcc-c++ + java-11-openjdk-devel + libxml2-devel + llvm-devel + llvm-static + ncurses-devel + openmotif + openmotif-devel + perl + perl-Digest-MD5 + udunits2 + udunits2-devel + which + zlib-devel + gtest-devel + python3-devel + python3-tkinter + xorg-x11-server-Xvfb +#-------- Ubuntu Only Dependencies ---------------- + - cfg: { os: ubuntu } + os_deps: >- + openjdk-11-jdk +#-------- Debian OS Only Dependencies ---------------- + - cfg: { os: debian } + os_deps: >- + openjdk-11-jdk +#-------- CentOS Only Dependencies ---------------- + - cfg: { os: centos } + pkg_mgr: yum + conf_pkg: yum -y install epel-release && yum -y update + os_deps: >- + libX11-devel + libXt-devel +#-------- Fedora Only Dependencies ---------------- +# - cfg: { os: fedora } +# pkg_mgr: dnf +# os_deps: >- +# swig +# perl-Text-Balanced +# python-devel +# diffutils +#-------- Version Specific Dependencies ---------------- + - cfg: { os: ubuntu, tag: 20.04 } + conf_pkg: DEBIAN_FRONTEND=noninteractive && apt-get update && apt-get install -y tzdata + install_gtest: cd /usr/src/gtest && cmake . && make && cp lib/libgtest* /usr/lib/ + tag_deps: >- + python3.8-dev + - cfg: { os: centos, tag: 7 } + tag_deps: >- + swig3 + python-devel + - cfg: { os: centos, tag: latest } + pkg_mgr: dnf + conf_pkg: > + dnf -y install epel-release && + dnf -y update && + dnf install -y 'dnf-command(config-manager)' && + dnf config-manager --enable powertools + tag_deps: >- + swig + python3-devel diffutils +#-------- Job definition ---------------- + runs-on: ubuntu-18.04 + container: docker://${{matrix.cfg.os}}:${{matrix.cfg.tag}} + steps: + # - name: Set up Python ${{ matrix.python-version }} + # uses: actions/setup-python@v2 + # with: + # python-version: ${{ matrix.python-version }} + - name: Info + run: | + pwd + echo $TEST + echo workspace is $TRICK_HOME + echo files in workspace: + ls -la $TRICK_HOME + env: + TEST: Hello World! + TRICK_HOME: ${{ github.workspace }} + - name: Update Package Manager + run: ${{matrix.conf_pkg}} + - name: Install Dependencies + run: | + ${{matrix.pkg_mgr}} ${{matrix.install_cmd}} ${{matrix.deps}} ${{matrix.arch_deps}} ${{matrix.os_deps}} ${{matrix.tag_deps}} + - name: Install GTest + run: ${{matrix.install_gtest}} + - name: Checkout repository + uses: actions/checkout@master + - name: Info after checkout + run: | + pwd + echo $TEST + echo workspace is $TRICK_HOME + echo files in workspace: + ls -la $TRICK_HOME + env: + TEST: Hello World! + TRICK_HOME: ${{ github.workspace }} + - name: Create testing environment + run: | + cd share/trick/pymods/trick/ + python3 -m venv .venv && . .venv/bin/activate && pip3 install -r requirements.txt + - name: Build trick + run: | + export MAKEFLAGS=-j`nproc` + ./configure + make + - name: Run Civet Tests + run: | + cd share/trick/pymods/trick/ + . .venv/bin/activate + ./run_tests.py + env: + TRICK_HOME: "${{ github.workspace }}" \ No newline at end of file diff --git a/.github/workflows/python_tests_macos.yml b/.github/workflows/python_tests_macos.yml new file mode 100644 index 00000000..1bb2fbab --- /dev/null +++ b/.github/workflows/python_tests_macos.yml @@ -0,0 +1,53 @@ +name: Python Tests MacOS + +on: + push: + paths-ignore: + - 'docs/**' + - '.github/workflows/**' + - '!.github/workflows/python_tests_macos.yml' + pull_request: + +jobs: + macOS: + runs-on: macos-latest + steps: + - name: Checkout repository + uses: actions/checkout@master + - name: Install python + run: | + brew install python + python3 -m ensurepip --upgrade + pip3 install virtualenv + python3 --version + pip3 --version + - name: Install gtest + run: | + wget https://github.com/google/googletest/archive/release-1.8.0.tar.gz + tar xzvf release-1.8.0.tar.gz + cd googletest-release-1.8.0/googletest + cmake . + make + make install + - name: Install dependencies + run: | + # sudo installer -pkg /Library/Developer/CommandLineTools/Packages/macOS_SDK_headers_for_macOS_10.15.pkg -target / + brew install --cask xquartz + brew install llvm@11 swig udunits openmotif maven python-tk + brew link llvm llvm@11 + - name: Create testing environment + run: | + cd share/trick/pymods/trick/ + python3 -m virtualenv .venv && . .venv/bin/activate && pip install -r requirements.txt + - name: Build Trick + run: | + export MAKEFLAGS=-j4 + ./configure + make + - name: Run Civet Tests + run: | + cd share/trick/pymods/trick/ + . .venv/bin/activate + ./run_tests.py + env: + TRICK_HOME: "${{ github.workspace }}" \ No newline at end of file diff --git a/.github/workflows/test_linux.yml b/.github/workflows/test_linux.yml index bf89de2f..28491c5c 100644 --- a/.github/workflows/test_linux.yml +++ b/.github/workflows/test_linux.yml @@ -142,4 +142,4 @@ jobs: ./configure make - name: Test - run: make test + run: make test \ No newline at end of file diff --git a/.gitignore b/.gitignore index 0735fad6..b7d0bea1 100644 --- a/.gitignore +++ b/.gitignore @@ -27,9 +27,11 @@ autom4te.cache trick_test gmon.out *init_log.csv* -include/mongoose/ -trick_source/web/HttpServer/mongoose.c -trick_source/web/HttpServer/mongoose.h -trick_source/web/HttpServer/obj/ +trick_source/web/CivetServer/obj/ trick-offline *sim_services_classes.resource +civetweb_clone/ +include/civet/ +.vscode/ +civet_server_error.log +server.pem \ No newline at end of file diff --git a/Makefile b/Makefile index be8bd0cd..3bb724bb 100644 --- a/Makefile +++ b/Makefile @@ -149,11 +149,10 @@ ifeq ($(USE_JAVA), 1) all: java endif -ifeq ($(TRICK_MONGOOSE), 1) -all: webserver -icg_sim_serv: ${TRICK_HOME}/include/mongoose/mongoose.h -ICG: ${TRICK_HOME}/include/mongoose/mongoose.h +ifeq ($(TRICK_CIVET), 1) +icg_sim_serv: ${TRICK_LIB_DIR}/libtrickCivet.a endif + #------------------------------------------------------------------------------- # 1.1 Build Trick-core no_dp: $(TRICK_LIB) $(TRICK_SWIG_LIB) @@ -219,47 +218,42 @@ dp: ${TRICK_HOME}/trick_source/trick_utils/units #------------------------------------------------------------------------------- # -.PHONY: webserver -webserver: ${TRICK_LIB_DIR}/libmongoose.a ${TRICK_HOME}/include/mongoose/mongoose.h - $(MAKE) -C ${TRICK_HOME}/trick_source/web/HttpServer #------------------------------------------------------------------------------- +# 1.2 Build Trick's CivetWeb webserver. -mongoose.h: - curl --retry 4 -O https://raw.githubusercontent.com/cesanta/mongoose/6.16/mongoose.h +CIVET_CLONE_DIR = civetweb_clone -mongoose.c: - curl --retry 4 -O https://raw.githubusercontent.com/cesanta/mongoose/6.16/mongoose.c +.PHONY: civetweb +civetweb: ${TRICK_LIB_DIR}/libtrickCivet.a -${TRICK_LIB_DIR}/libmongoose.a: ${TRICK_HOME}/include/mongoose/mongoose.h | mongoose.o $(TRICK_LIB_DIR) - ar crs $@ mongoose.o - @ rm mongoose.o - @ rm -f mongoose.h - @ echo ; echo "Mongoose library compiled:" ; date +${TRICK_LIB_DIR}/libtrickCivet.a: ${TRICK_LIB_DIR}/libcivetweb.a ${TRICK_HOME}/include/civet/civetweb.h ${TRICK_HOME}/include/civet/CivetServer.h + $(MAKE) -C ${TRICK_HOME}/trick_source/web/CivetServer -ifeq (${TRICK_OFFLINE}, 0) +${TRICK_LIB_DIR}/libcivetweb.a: ${CIVET_CLONE_DIR}/libcivetweb.a | ${TRICK_LIB_DIR} + cp ${CIVET_CLONE_DIR}/libcivetweb.a $(TRICK_LIB_DIR)/libcivetweb.a -mongoose.o: mongoose.h mongoose.c - $(CC) $(TRICK_CFLAGS) ${TRICK_SYSTEM_CXXFLAGS} -c -o mongoose.o mongoose.c - @ rm mongoose.c +${TRICK_HOME}/include/civet: + mkdir -p ${TRICK_HOME}/include/civet -${TRICK_HOME}/include/mongoose/mongoose.h: mongoose.h | ${TRICK_HOME}/include/mongoose - @ cp mongoose.h $@ +${TRICK_HOME}/include/civet/civetweb.h: ${CIVET_CLONE_DIR} ${TRICK_HOME}/include/civet + cp ${CIVET_CLONE_DIR}/include/civetweb.h ${TRICK_HOME}/include/civet/civetweb.h +${TRICK_HOME}/include/civet/CivetServer.h: ${CIVET_CLONE_DIR} ${TRICK_HOME}/include/civet + cp ${CIVET_CLONE_DIR}/include/CivetServer.h ${TRICK_HOME}/include/civet/CivetServer.h + + +ifeq (${TRICK_FORCE_32BIT},1) +CIVET_COMPILE_FAGS=-m32 else - -# if trick-offline gets updated, we should rebuild libmongoose -${TRICK_LIB_DIR}/libmongoose.a: ${TRICK_HOME}/trick-offline/mongoose.h ${TRICK_HOME}/trick-offline/mongoose.c - -mongoose.o: ${TRICK_HOME}/trick-offline/mongoose.h ${TRICK_HOME}/trick-offline/mongoose.c - $(CC) $(TRICK_CFLAGS) -c -I${TRICK_HOME}/trick-offline -o mongoose.o ${TRICK_HOME}/trick-offline/mongoose.c - -${TRICK_HOME}/include/mongoose/mongoose.h: ${TRICK_HOME}/trick-offline/mongoose.h | ${TRICK_HOME}/include/mongoose - @ cp ${TRICK_HOME}/trick-offline/mongoose.h $@ +CIVET_COMPILE_FAGS= endif -${TRICK_HOME}/include/mongoose: - @ mkdir $@ +${CIVET_CLONE_DIR}/libcivetweb.a: ${CIVET_CLONE_DIR} + $(MAKE) -C ${CIVET_CLONE_DIR} lib COPT=${CIVET_COMPILE_FAGS} WITH_CPP=1 WITH_WEBSOCKET=1 + +${CIVET_CLONE_DIR}: + git clone --branch v1.14 --depth 1 https://github.com/civetweb/civetweb.git $@ #------------------------------------------------------------------------------- # 1.3 Build Trick's Java Tools @@ -338,6 +332,10 @@ sim_test: @ $(MAKE) -C test @ $(MAKE) -C trick_sims test +pytest: + make -C share/trick/pymods/trick + + #requirements: # @ $(MAKE) -C trick_test/requirements_docs install diff --git a/autoconf/configure.ac b/autoconf/configure.ac index b5a5a262..00cd22ad 100644 --- a/autoconf/configure.ac +++ b/autoconf/configure.ac @@ -345,22 +345,6 @@ AC_ARG_ENABLE([offline], ) AC_SUBST([TRICK_OFFLINE]) -# If offline is specified, set some compilation flags -AC_ARG_ENABLE([mongoose], - AS_HELP_STRING([--enable-mongoose], [Compile Trick with webserver capabilites.]), - AS_IF([test "x$enable_mongoose" = xyes], - [ - TRICK_MONGOOSE="1" - AC_MSG_WARN($(tput setaf 1) --enable-mongoose Mongoose is released under GPLv2 and Trick is \ -released under NASA Open Source Agreement 1.3. Distribution must comply with \ -these agreements$(tput sgr0)) - ], - [TRICK_MONGOOSE="0"] - ), - [TRICK_MONGOOSE="0"] -) -AC_SUBST([TRICK_MONGOOSE]) - # If offline is specified, set some compilation flags AC_ARG_ENABLE([offline], AS_HELP_STRING([--enable-offline], [Compile Trick in offline mode. Un-tar the trick-offline directory in TRICK_HOME to use this feature]), @@ -372,6 +356,27 @@ AC_ARG_ENABLE([offline], ) AC_SUBST([TRICK_OFFLINE]) +AC_ARG_ENABLE([civet], + AS_HELP_STRING([--enable-civet], [Compile Trick with webserver capabilites.]), + AS_IF([test "x$enable_civet" = xyes], + [ + TRICK_DISABLE_CIVET="0" + ], + [TRICK_DISABLE_CIVET="1"] + ), + [TRICK_DISABLE_CIVET="0"] +) +AC_SUBST([TRICK_DISABLE_CIVET]) + +# If offline is specified, set some compilation flags +AC_ARG_ENABLE([offline], + AS_HELP_STRING([--enable-offline], [Compile Trick in offline mode. Un-tar the trick-offline directory in TRICK_HOME to use this feature]), + AS_IF([test "x$enable_offline" = xyes], + [TRICK_DISABLE_CIVET="1"], + ), +) +AC_SUBST([TRICK_OFFLINE]) + dnl look for programs we need to compile and run AC_PROG_AWK AC_PROG_CC diff --git a/configure b/configure index 046629d9..e92782f2 100755 --- a/configure +++ b/configure @@ -671,7 +671,7 @@ ac_ct_CC CFLAGS CC AWK -TRICK_MONGOOSE +TRICK_DISABLE_CIVET TRICK_OFFLINE TRICK_FORCE_32BIT LIBXML @@ -717,6 +717,7 @@ infodir docdir oldincludedir includedir +runstatedir localstatedir sharedstatedir sysconfdir @@ -742,7 +743,7 @@ enable_option_checking with_x enable_32bit enable_offline -enable_mongoose +enable_civet with_python with_prepend_path with_swig @@ -812,6 +813,7 @@ datadir='${datarootdir}' sysconfdir='${prefix}/etc' sharedstatedir='${prefix}/com' localstatedir='${prefix}/var' +runstatedir='${localstatedir}/run' includedir='${prefix}/include' oldincludedir='/usr/include' docdir='${datarootdir}/doc/${PACKAGE_TARNAME}' @@ -1064,6 +1066,15 @@ do | -silent | --silent | --silen | --sile | --sil) silent=yes ;; + -runstatedir | --runstatedir | --runstatedi | --runstated \ + | --runstate | --runstat | --runsta | --runst | --runs \ + | --run | --ru | --r) + ac_prev=runstatedir ;; + -runstatedir=* | --runstatedir=* | --runstatedi=* | --runstated=* \ + | --runstate=* | --runstat=* | --runsta=* | --runst=* | --runs=* \ + | --run=* | --ru=* | --r=*) + runstatedir=$ac_optarg ;; + -sbindir | --sbindir | --sbindi | --sbind | --sbin | --sbi | --sb) ac_prev=sbindir ;; -sbindir=* | --sbindir=* | --sbindi=* | --sbind=* | --sbin=* \ @@ -1201,7 +1212,7 @@ fi for ac_var in exec_prefix prefix bindir sbindir libexecdir datarootdir \ datadir sysconfdir sharedstatedir localstatedir includedir \ oldincludedir docdir infodir htmldir dvidir pdfdir psdir \ - libdir localedir mandir + libdir localedir mandir runstatedir do eval ac_val=\$$ac_var # Remove trailing slashes. @@ -1354,6 +1365,7 @@ Fine tuning of the installation directories: --sysconfdir=DIR read-only single-machine data [PREFIX/etc] --sharedstatedir=DIR modifiable architecture-independent data [PREFIX/com] --localstatedir=DIR modifiable single-machine data [PREFIX/var] + --runstatedir=DIR modifiable per-process data [LOCALSTATEDIR/run] --libdir=DIR object code libraries [EPREFIX/lib] --includedir=DIR C header files [PREFIX/include] --oldincludedir=DIR C header files for non-gcc [/usr/include] @@ -1395,7 +1407,7 @@ Optional Features: --enable-offline Compile Trick in offline mode. Un-tar the trick-offline directory in TRICK_HOME to use this feature - --enable-mongoose Compile Trick with webserver capabilites. + --enable-civet Compile Trick with webserver capabilites. --enable-java use java (default is yes) --enable-er7utils use er7_utils (default is yes) @@ -3875,30 +3887,6 @@ fi -# If offline is specified, set some compilation flags -# Check whether --enable-mongoose was given. -if test "${enable_mongoose+set}" = set; then : - enableval=$enable_mongoose; if test "x$enable_mongoose" = xyes; then : - - TRICK_MONGOOSE="1" - { $as_echo "$as_me:${as_lineno-$LINENO}: WARNING: $(tput setaf 1) --enable-mongoose Mongoose is released under GPLv2 and Trick is \ -released under NASA Open Source Agreement 1.3. Distribution must comply with \ -these agreements$(tput sgr0)" >&5 -$as_echo "$as_me: WARNING: $(tput setaf 1) --enable-mongoose Mongoose is released under GPLv2 and Trick is \ -released under NASA Open Source Agreement 1.3. Distribution must comply with \ -these agreements$(tput sgr0)" >&2;} - -else - TRICK_MONGOOSE="0" - -fi -else - TRICK_MONGOOSE="0" - -fi - - - # If offline is specified, set some compilation flags # Check whether --enable-offline was given. if test "${enable_offline+set}" = set; then : @@ -3915,6 +3903,33 @@ fi +# Check whether --enable-civet was given. +if test "${enable_civet+set}" = set; then : + enableval=$enable_civet; if test "x$enable_civet" = xyes; then : + + TRICK_DISABLE_CIVET="0" + +else + TRICK_DISABLE_CIVET="1" + +fi +else + TRICK_DISABLE_CIVET="0" + +fi + + + +# If offline is specified, set some compilation flags +# Check whether --enable-offline was given. +if test "${enable_offline+set}" = set; then : + enableval=$enable_offline; if test "x$enable_offline" = xyes; then : + TRICK_DISABLE_CIVET="1" +fi +fi + + + for ac_prog in gawk mawk nawk awk do # Extract the first word of "$ac_prog", so it can be a program name with args. diff --git a/docs/documentation/web/Adding_SSL_encryption_to_Your_Webserver.md b/docs/documentation/web/Adding_SSL_encryption_to_Your_Webserver.md new file mode 100644 index 00000000..e4808818 --- /dev/null +++ b/docs/documentation/web/Adding_SSL_encryption_to_Your_Webserver.md @@ -0,0 +1,36 @@ +# Adding SSL Encryption to Your Webserver + +## Getting started ++ Install OpenSSL on your system. There are OpenSSL install packages for all major Linux distributions. ++ Create a SSL certificate ++ Edit your input file + +## Creating a SSL certificate +You can use your own certificate but to create a self signed certificate follow the steps below: +``` + openssl genrsa -des3 -out server.key 1024 + + openssl req -new -key server.key -out server.csr + + cp server.key server.key.orig + + openssl rsa -in server.key.orig -out server.key + + openssl x509 -req -days 3650 -in server.csr -signkey server.key -out server.crt + + cp server.crt server.pem + + cat server.key >> server.pem +``` + +## Edit your input file + +add the following lines to your input file +```python +web.server.ssl_enable = True +web.server.path_to_ssl_cert = "server.pem" +``` +Where server.pem is the path to the server.pem file you created when creating a SSL certificate + +## Access your webserver +Now that ssl encryption is enabled, to access you webserver use https://localhost.ssl:8888 instead of http and wss://localhost.ssl:8888 for http and ws protocals respectively. diff --git a/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.html b/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.html deleted file mode 100644 index 1ccf86de..00000000 --- a/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.html +++ /dev/null @@ -1,558 +0,0 @@ - - - - - - - -Adding_a_Web_Server_to_Your_Sim - - - - - - - - - - - -

Adding a Web Server to Your Sim

- -

To add a web server to your simulation, simply include the WebServer sim module into your S_define file:

- -
#include "sim_objects/WebServer.sm"
- -

Configuration of the Web Server

- -

The following (input.py) parameters are available to configure your web server:

- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
Parameter NameDefault ValueDescription
web.server.enableFalseMust be explicitly enabled
web.server.port"8888"Web servers “listen” port
web.server.document_root"www"Web servers document root
web.server.debugFalsePrint Client/Server Communication.
- -

For your web server to be active, you must at least specify the following :

- -
web.server.enable = True
-
- -

To have your web server listen on port 8890, rather than 8888, you would specify:

- -
web.server.port = "8890"
- -

To serve files from a directory called my_document_root, rather than www:

- -
web.server.document_root = "my_document_root"
- -

To see client/server communication:

- -
web.server.debug = True
- -

When the Web Server Starts

- -

The web server, if enabled, will start during sim initialization. When it does, it will look for the specified document root directory. By default that’s “www”. If root directory doesn’t exist, one will be created with a simple index.html file , a style sheet, and a couple of directories.

- -

Connecting to Your Web Server

- -

Assuming that you accepted the default port, connect to http://localhost:8888/ from your web browser. This will display the index.html file in your root directory.

- -

The Default Document Root Directory

- -

The default document root directory that was initially created for you is minimal.

- -
www/
-    index.html
-    style.css
-    apps/
-    images/
- -

index.html is the file that’s displayed when you connect to http://localhost:8888/.

- -

style.css is a CSS style-sheet that’s included by index.html to give it some pizzazz.

- -

The apps directory contains links to some example html/javascript applications - in $TRICK_HOME/trick_source/web/apps.

- -

The images directory contains trick_icon.png.

- -

You are encouraged to add to, modify, and/or delete these files and directories to best suite the needs of your project.

- - - - - - - - - - - diff --git a/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.md b/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.md index 3cb78ee6..478dcaf3 100644 --- a/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.md +++ b/docs/documentation/web/Adding_a_Web_Server_to_Your_Sim.md @@ -1,30 +1,33 @@ -# Adding a Web Server to Your Sim +# Adding a Web Server to Your Sim -To add a web server to your simulation, simply include the WebServer sim module into your **S_define** file: +To add a web server to your simulation, simply include the CivetServer sim module into your **S_define** file: -``` -#include "sim_objects/WebServer.sm" +``` +#include "sim_objects/CivetServer.sm" ``` -## Configuration of the Web Server +## Configuration of the Web Server -The following (input.py) parameters are available to configure your web server: +The following (input.py) parameters are available to configure your web server: -|Parameter Name | Default Value| Description | -|------------------------|--------------|----------------------------------| -|web.server.enable | False |Must be explicitly enabled | -|web.server.port | "8888" |Web servers “listen” port | -|web.server.document_root| "www" |Web servers document root | -|web.server.debug | False |Print Client/Server Communication.| +|Parameter Name | Default Value | Description | +|---------------------------|---------------------------|-----------------------------------------------------------------| +|web.server.enable | False |Must be explicitly enabled | +|web.server.port | "8888" |Web servers “listen” port | +|web.server.document_root | "www" |Web servers document root | +|web.server.debug | False |Print Client/Server Communication. | +|web.server.ssl_enable | False |Encrypt traffic. Uses https instead of http. | +|web.server.path_to_ssl_cert|"~/.ssl/server.pem" |Path to your certificate. This is only used if ssl_enable = True| +|web.server.error_log_file | "civet_server_error.log" |CivetWeb error log file. | -For your web server to be active, you must at least specify the following : +For your web server to be active, you must at least specify the following : ```python -web.server.enable = True +web.server.enable = True ``` -To have your web server listen on port 8890, rather than 8888, you would specify: +To have your web server listen on port 8890, rather than 8888, you would specify: ```python web.server.port = "8890" @@ -32,8 +35,8 @@ web.server.port = "8890" To serve files from a directory called ```my_document_root```, rather than ```www```: -```python -web.server.document_root = "my_document_root" +```python +web.server.document_root = "my_document_root" ``` To see client/server communication: @@ -42,34 +45,34 @@ To see client/server communication: web.server.debug = True ``` -## When the Web Server Starts -The web server, if enabled, will start during sim initialization. When it does, it will look for the specified document root directory. By default that’s “www”. If root directory doesn’t exist, one will be created with a simple index.html file , a style sheet, and a couple of directories. +## When the Web Server Starts +The web server, if enabled, will start during sim initialization. When it does, it will look for the specified document root directory. By default that’s “www”. If root directory doesn’t exist, one will be created with a simple index.html file , a style sheet, and a couple of directories. -## Connecting to Your Web Server -Assuming that you accepted the default port, connect to ```http://localhost:8888/``` from your web browser. This will display the index.html file in your root directory. +## Connecting to Your Web Server +Assuming that you accepted the default port, connect to ```http://localhost:8888/``` (```https://localhost:8888/``` if ssl_enable=True) from your web browser. This will display the index.html file in your root directory. ## The Default Document Root Directory -The default document root directory that was initially created for you is minimal. +The default document root directory that was initially created for you is minimal. ``` -www/ - index.html - style.css - apps/ - images/ -``` +www/ + index.html + style.css + apps/ + images/ +``` -**index.html** is the file that’s displayed when you connect to http://localhost:8888/. +**index.html** is the file that’s displayed when you connect to http://localhost:8888/. -**style.css** is a CSS style-sheet that’s included by index.html to give it some pizzazz. +**style.css** is a CSS style-sheet that’s included by index.html to give it some pizzazz. -The **apps** directory contains links to some example html/javascript applications - in ```$TRICK_HOME/trick_source/web/apps```. +The **apps** directory contains links to some example html/javascript applications + in ```$TRICK_HOME/trick_source/web/apps```. -The **images** directory contains trick_icon.png. +The **images** directory contains trick_icon.png. -**You are encouraged to add to, modify, and/or delete these files and directories to best suite the needs of your project.** +**You are encouraged to add to, modify, and/or delete these files and directories to best suite the needs of your project.** diff --git a/docs/documentation/web/Extending_the_HTTP-API.html b/docs/documentation/web/Extending_the_HTTP-API.html deleted file mode 100644 index b2bd7d57..00000000 --- a/docs/documentation/web/Extending_the_HTTP-API.html +++ /dev/null @@ -1,583 +0,0 @@ - - - - - - - -Extending_the_HTTP-API - - - - - - - - - - - -

Extending the HTTP-API

- -

The HTTP-API is implemented as a collection of httpMethodHandlers. An httpMethodHandler is a pointer to a function that is expected to respond to an HTTP GET request, using the Cesanta Mongoose framework. An httpMethodHandler is defined (in trick/WebServer.hh) as follows:

- -
typedef void (*httpMethodHandler)(struct mg_connection*, struct http_message*);
- -

Documentation for the Cesanta Mongoose Networking Library can be found at: -https://cesanta.com/docs/overview/intro.html

- -

Example HTTP-API Extension

- -

Suppose you want your web server to send you a JSON message:

- -
{ "greeting" : "Hello Trick Sim Developer!" }
- -

when you invoke the URL: http://localhost:8888/api/http/hello.

- -

Creating an httpMethodHandler.

- -

The following two files will be our implementation of an httpMethodHandler. We'll put these in some models directory httpMethods/.

- -

handle_HTTP_GET_hello.h

- -
#ifndef HANDLE_HTTP_GET_HELLO
-#define HANDLE_HTTP_GET_HELLO
-
-#ifndef SWIG
-void handle_HTTP_GET_hello(struct mg_connection *nc, struct http_message *hm);
-#endif
-
-#endif
- -

handle_HTTP_GET_hello.c

- -
#include "mongoose/mongoose.h"
-
-void handle_HTTP_GET_hello(struct mg_connection *nc, struct http_message *hm) {
-    mg_printf(nc, "%s", "HTTP/1.1 200 OK\r\nTransfer-Encoding: chunked\r\n\r\n");
-    const char* json_text =
-        "{ \"greeting\" : \"Hello Trick Sim Developer!\" }";
-    mg_printf_http_chunk(nc, "%s", json_text);
-    mg_send_http_chunk(nc, "", 0);
-}
- -

Installing our httpMethodHandler.

- -

We'll do this from our S_define file:

- - - - - -
web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello );
- -

A Complete S_define

- -
/***********************TRICK HEADER*************************
-PURPOSE:
-    (Cannon Numeric)
-LIBRARY DEPENDENCIES:
-    (
-     (cannon/gravity/src/cannon_init.c)
-     (cannon/gravity/src/cannon_numeric.c)
-     (httpMethods/handle_HTTP_GET_hello.c)
-    )
-*************************************************************/
-
-#include "sim_objects/default_trick_sys.sm"
-#include "sim_objects/WebServer.sm"
-##include "cannon/gravity/include/cannon_numeric.h"
-##include "httpMethods/handle_HTTP_GET_hello.h"
-
-class CannonSimObject : public Trick::SimObject {
-
-    public:
-        CANNON      cannon ;
-        int foo;
-        CannonSimObject() {
-            ("default_data") cannon_default_data( &cannon ) ;
-            ("initialization") cannon_init( &cannon ) ;
-            ("derivative") cannon_deriv( &cannon ) ;
-            ("integration") trick_ret = cannon_integ( &cannon ) ;
-            ("dynamic_event") cannon_impact( &cannon) ;
-        }
-} ;
-CannonSimObject dyn ;
-
-IntegLoop dyn_integloop (0.01) dyn;
-
-void create_connections() {
-    dyn_integloop.getIntegrator(Runge_Kutta_4, 5);
-    web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello );
-}
-
- - - - - - - - - - - - - - - - - diff --git a/docs/documentation/web/Extending_the_HTTP-API.md b/docs/documentation/web/Extending_the_HTTP-API.md index 53cc69bd..8e82078c 100644 --- a/docs/documentation/web/Extending_the_HTTP-API.md +++ b/docs/documentation/web/Extending_the_HTTP-API.md @@ -1,77 +1,79 @@ -##Extending the HTTP-API - -The HTTP-API is implemented as a collection of ```httpMethodHandlers```. An ```httpMethodHandler``` is a pointer to a function that is expected to respond to an HTTP GET request, using the **Cesanta Mongoose** framework. An ```httpMethodHandler``` is defined (in ```trick/WebServer.hh```) as follows: - - - - +## Extending the HTTP-API -```c -typedef void (*httpMethodHandler)(struct mg_connection*, struct http_message*); -``` - -Documentation for the **Cesanta Mongoose Networking Library** can be found at: -[https://cesanta.com/docs/overview/intro.html](https://cesanta.com/docs/overview/intro.html) - -## Example HTTP-API Extension - -Suppose you want your web server to send you a JSON message: - -```json -{ "greeting" : "Hello Trick Sim Developer!" } -``` - -when you invoke the URL: ```http://localhost:8888/api/http/hello```. - -### Creating an ```httpMethodHandler```. - -The following two files will be our implementation of an ```httpMethodHandler```. We'll put these in some models directory ```httpMethods/```. - -**```handle_HTTP_GET_hello.h```** +The HTTP-API is implemented as a collection of ```httpMethodHandlers```. An ```httpMethodHandler``` is a pointer to a function that is expected to respond to an HTTP GET request, using the **CivetWeb** framework. An ```httpMethodHandler``` is defined (in ```trick/CivetWeb.hh```) as follows: + + + + + +```c +typedef void (*httpMethodHandler)(struct mg_connection *, void* cbdata); +``` + +Documentation for the **CivetWeb Networking Library** can be found at: +[https://cesanta.com/docs/overview/intro.html](http://civetweb.github.io/civetweb/) + +## Example HTTP-API Extension + +Suppose you want your web server to send you a JSON message: + +```json +{ "greeting" : "Hello Trick Sim Developer!" } +``` + +when you invoke the URL: ```http://localhost:8888/api/http/hello```. + +### Creating an ```httpMethodHandler```. + +The following two files will be our implementation of an ```httpMethodHandler```. We'll put these in some models directory ```httpMethods/```. + +**```handle_HTTP_GET_hello.h```** ```c #ifndef HANDLE_HTTP_GET_HELLO #define HANDLE_HTTP_GET_HELLO #ifndef SWIG -void handle_HTTP_GET_hello(struct mg_connection *nc, struct http_message *hm); +void handle_HTTP_GET_hello(struct mg_connection *nc, void *hm); #endif -#endif +#endif ``` - -**```handle_HTTP_GET_hello.c```** - -```c -#include "mongoose/mongoose.h" -void handle_HTTP_GET_hello(struct mg_connection *nc, struct http_message *hm) { +**```handle_HTTP_GET_hello.c```** + +```c +#include "civet/CivetServer.h" +#include "civet/civetweb.h" +#include + +void handle_HTTP_GET_hello(struct mg_connection *nc, void *hm) { mg_printf(nc, "%s", "HTTP/1.1 200 OK\r\nTransfer-Encoding: chunked\r\n\r\n"); const char* json_text = "{ \"greeting\" : \"Hello Trick Sim Developer!\" }"; - mg_printf_http_chunk(nc, "%s", json_text); - mg_send_http_chunk(nc, "", 0); -} -``` - -### Installing our ```httpMethodHandler```. - -We'll do this from our **S_define** file: - -* Add ```(httpMethods/handle_HTTP_GET_hello.c)``` to the ```LIBRARY DEPENDENCIES```. - -* Include our header file: - - ```##include "httpMethods/handle_HTTP_GET_hello.h"``` - -* In ```create_connections()``` add : - -```c -web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello ); -``` -### A Complete S_define - -```c++ + mg_send_chunk(nc, json_text, strlen(json_text)); + mg_send_chunk(nc, "", 0); +} +``` + +### Installing our ```httpMethodHandler```. + +We'll do this from our **S_define** file: + +* Add ```(httpMethods/handle_HTTP_GET_hello.c)``` to the ```LIBRARY DEPENDENCIES```. + +* Include our header file: + + ```##include "httpMethods/handle_HTTP_GET_hello.h"``` + +* In ```create_connections()``` add : + +```c +web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello ); +``` +### A Complete S_define + +```c++ /***********************TRICK HEADER************************* PURPOSE: (Cannon Numeric) @@ -84,7 +86,7 @@ LIBRARY DEPENDENCIES: *************************************************************/ #include "sim_objects/default_trick_sys.sm" -#include "sim_objects/WebServer.sm" +#include "sim_objects/CivetServer.sm" ##include "cannon/gravity/include/cannon_numeric.h" ##include "httpMethods/handle_HTTP_GET_hello.h" @@ -109,5 +111,5 @@ void create_connections() { dyn_integloop.getIntegrator(Runge_Kutta_4, 5); web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello ); } - -``` + +``` diff --git a/docs/documentation/web/Extending_the_WS-API.html b/docs/documentation/web/Extending_the_WS-API.html deleted file mode 100644 index c91b9ee9..00000000 --- a/docs/documentation/web/Extending_the_WS-API.html +++ /dev/null @@ -1,736 +0,0 @@ - - - - - - - -Extending_the_WS-API - - - - - - - - - - - -

Extending the WebSocket-API

- -

When You Create a WebSocket Connection

- -

Consider the following Javascript, that creates a web socket connection:

- -

var ws = new WebSocket('ws://localhost:8888/api/ws/VariableServer');

- -

In the URL: ws://localhost:8888/api/ws/VariableServer

- - - -

In the Trick web server, the path associated with a websocket must begin with -/api/ws/. The remaining part of the path, i.e., VariableServer is the key that specifies the sub-protocol, prescribing what messages will be passed between client and server, and what those messages mean.

- -

When a web-socket connection is established, the key will determine what type (sub-class) of WebSocketSession object to create, to manage the connection.

- -

WebSocketSession

- -

A WebSocketSession is a pure virtual base class meant to represent the state of one of potentially many websocket connections. It provides methods to:

- -
    -
  1. Synchronously marshall Trick simulation data for out-going messages
  2. -
  3. Send messages to the websocket client, and
  4. -
  5. Receive and process messages from the websocket client.
  6. -
- -

To implement a new websocket sub-protocol, one needs to derive a new class from this base class, and implement the required methods. WebSocketSession.hh can be found in ${TRICK_HOME}/include/trick/.

- -

WebSocketSession.hh

- -
/*************************************************************************
-PURPOSE: (Represent Websocket connection.)
-**************************************************************************/
-#ifndef WEB_SOCKET_SESSION_HH
-#define WEB_SOCKET_SESSION_HH
-
-#include <string>
-#ifndef SWIG
-#include "mongoose/mongoose.h"
-#endif
-
-class WebSocketSession {
-    public:
-        WebSocketSession(struct mg_connection *nc):connection(nc){};
-        virtual ~WebSocketSession() {};
-        virtual void marshallData()=0;
-        virtual void sendMessage()=0;
-        virtual int  handleMessage(std::string)=0;
-
-        struct mg_connection* connection;
-};
-#endif
- -

Adding Your New WebSocketSession Type to the WebServer

- -

To install your new websocket protocol, you'll need to create a function that -creates an instance of your new WebSocketSession type. Then you'll need to call -HTTP_Server::installWebSocketSessionMaker to install the function, with a -label.

- -

The function you'll create will take struct mg_connection * as an argument -and return WebSocketSession*.

- -

Example

- -

Let's create a new web socket protocol that sends the time in GMT or local time.

- -

First we'll derive a new type called TimeSession from WebSocketSession.

- -

TimeSession.hh

- -
/*************************************************************************
-PURPOSE: (Represent the state of a variable server websocket connection.)
-**************************************************************************/
-#ifndef TIMESESSION_HH
-#define TIMESESSION_HH
-#include <vector>
-#include <string>
-#include "time.h"
-#include "trick/WebSocketSession.hh"
-
-class TimeSession : public WebSocketSession {
-    public:
-        enum Zone { GMT, LOCAL};
-        TimeSession(struct mg_connection *nc);
-        ~TimeSession();
-        void marshallData();
-        void sendMessage();
-        int  handleMessage(std::string);
-    private:
-        time_t now;
-        Zone zone;
-};
-
-WebSocketSession* makeTimeSession( struct mg_connection *nc );
-#endif
- -

Below is our implementation. Notice the function makeTimeSession at the bottom.

- -

TimeSession.cpp

- -
#include <stdio.h>
-#include <time.h>
-#include <iostream>
-#include "TimeSession.hh"
-
-// CONSTRUCTOR
-TimeSession::TimeSession( struct mg_connection *nc ) : WebSocketSession(nc) {
-    time(&now);
-}
-
-// DESTRUCTOR
-TimeSession::~TimeSession() {}
-
-void TimeSession::marshallData() {
-    time(&now);
-}
-
-void TimeSession::sendMessage() {
-
-    char message[1024];
-    struct tm *theTime;
-    if (zone == TimeSession::LOCAL) {
-        theTime = localtime(&now);
-    } else {
-        theTime = gmtime(&now);
-    }
-    int hours   = theTime->tm_hour;
-    int minutes = theTime->tm_min;
-    int seconds = theTime->tm_sec;
-    int day     = theTime->tm_mday;
-    int month   = theTime->tm_mon + 1;
-    int year    = theTime->tm_year + 1900;
-
-    sprintf(message, "Time: %02d:%02d:%02d Date: %02d/%02d/%d\n", hours, minutes, seconds, month, day, year);
-    mg_send_websocket_frame(connection, WEBSOCKET_OP_TEXT, message, strlen(message));
-}
-
-int TimeSession::handleMessage(std::string client_msg) {
-
-   if (client_msg.compare("GMT") == 0) {
-       zone = TimeSession::GMT;
-   } else if (client_msg.compare("LOCAL") == 0) {
-       zone = TimeSession::LOCAL;
-   } else {
-      std::cerr << "ERROR: Unknown command \"" << client_msg << "\"." << std::endl;
-   }
-   return 0;
-}
-
-// WebSocketSessionMaker function for a TimeSession.
-WebSocketSession* makeTimeSession( struct mg_connection *nc ) {
-    std::cerr << "DEBUG: Creating new TimeSession." << std::endl;
-    return new TimeSession(nc);
-}
- -

We put TimeSession.cpp and TimeSession.cpp into a models directory called httpMethods/.

- -

S_define Modifications

- -
    -
  1. Specify the dependency on the httpMethods/TimeSession.cpp compilation unit.
  2. -
  3. We should already be including the WebServer sim object, otherwise we don't even have a webserver.
  4. -
  5. We need to include our new header file: ##include "httpMethods/TimeSession.hh"
  6. -
  7. Finally, install our WebSocketSession type: web.server.installWebSocketSessionMaker("Time", &makeTimeSession); -The label we use for our protocol here is "Time", but it can be whatever name you choose.
  8. -
- -
/***********************TRICK HEADER*************************
-PURPOSE:
-    (Cannon Numeric)
-LIBRARY DEPENDENCIES:
-    (
-     (cannon/gravity/src/cannon_init.c)
-     (cannon/gravity/src/cannon_numeric.c)
-     (httpMethods/TimeSession.cpp)                                    // <--(1)
-    )
-*************************************************************/
-
-#include "sim_objects/default_trick_sys.sm"
-#include "sim_objects/WebServer.sm"                                   // <--(2)
-##include "cannon/gravity/include/cannon_numeric.h"
-##include "httpMethods/TimeSession.hh"                                // <--(3)
-
-class CannonSimObject : public Trick::SimObject {
-
-    public:
-        CANNON      cannon ;
-        int foo;
-        CannonSimObject() {
-            ("default_data") cannon_default_data( &cannon ) ;
-            ("initialization") cannon_init( &cannon ) ;
-            ("derivative") cannon_deriv( &cannon ) ;
-            ("integration") trick_ret = cannon_integ( &cannon ) ;
-            ("dynamic_event") cannon_impact( &cannon) ;
-        }
-} ;
-CannonSimObject dyn ;
-
-IntegLoop dyn_integloop (0.10) dyn;
-
-void create_connections() {
-    dyn_integloop.getIntegrator(Runge_Kutta_4, 5);
-    web.server.installWebSocketSessionMaker("Time", &makeTimeSession);  // <--(4)
-}
- -

Testing The New WebSocket Interface

- -

To test your new web socket interface, put the following time.html file in $YOUR_SIM_DIRECTORY/www/apps. Then request http://localhost:8888/apps/time.html from your browser. You should see the time messages from your sim.

- -

time.html

- -
<!DOCTYPE html>
-<html>
-  <head>
-    <title>WS Example</title>
-  </head>
-  <body>
-      <div id="output"></div>
-      <script type="text/javascript">
-
-          function log(s) {
-              var p = document.createElement("p");
-              p.style.wordWrap = "break-word";
-              p.textContent = s;
-              output.appendChild(p);
-          }
-
-          var ws = new WebSocket('ws://localhost:8888/api/ws/Time');
-
-          // WebSocket Event Handlers
-          ws.onopen = function(e) {
-              ws.send("GMT");
-          };
-          ws.onmessage = function(e) {
-             log(e.data);
-          };
-          ws.onerror = function(e) {
-             console.log("WebSocket Error: " , e);
-             handleErrors(e);
-          };
-          ws.onclose = function(e) {
-             console.log("Connection closed", e);
-          };
-
-      </script>
-  </body>
-</html>
- - - - - - - - - - - - - - - - - diff --git a/docs/documentation/web/Extending_the_WS-API.md b/docs/documentation/web/Extending_the_WS-API.md index 7faf00f5..21da130d 100644 --- a/docs/documentation/web/Extending_the_WS-API.md +++ b/docs/documentation/web/Extending_the_WS-API.md @@ -1,4 +1,4 @@ -#Extending the WebSocket-API +## Extending the WebSocket-API ## When You Create a WebSocket Connection @@ -23,9 +23,9 @@ A ```WebSocketSession``` is a pure virtual base class meant to represent the sta 1. Synchronously marshall Trick simulation data for out-going messages 2. Send messages to the websocket client, and - 3. Receive and process messages from the websocket client. - -To implement a new websocket sub-protocol, one needs to derive a new class from this base class, and implement the required methods. ```WebSocketSession.hh``` can be found in ```${TRICK_HOME}/include/trick/```. + 3. Receive and process messages from the websocket client. + +To implement a new websocket sub-protocol, one needs to derive a new class from this base class, and implement the required methods. ```WebSocketSession.hh``` can be found in ```${TRICK_HOME}/include/trick/```. ### WebSocketSession.hh ```c @@ -37,13 +37,19 @@ PURPOSE: (Represent Websocket connection.) #include #ifndef SWIG -#include "mongoose/mongoose.h" +#include "civet/CivetServer.h" #endif class WebSocketSession { public: WebSocketSession(struct mg_connection *nc):connection(nc){}; virtual ~WebSocketSession() {}; + + /** + When HTTP_Server::time_homogeneous is set, WebSocketSession::marshallData() is called from the main + sim thread in a "top_of_frame" job, so that all of the data can be staged at + the same sim-time, in other words it's time-homogeneous. + */ virtual void marshallData()=0; virtual void sendMessage()=0; virtual int handleMessage(std::string)=0; @@ -109,6 +115,7 @@ Below is our implementation. Notice the function ```makeTimeSession``` at the bo #include #include #include "TimeSession.hh" +#include // CONSTRUCTOR TimeSession::TimeSession( struct mg_connection *nc ) : WebSocketSession(nc) { @@ -139,7 +146,7 @@ void TimeSession::sendMessage() { int year = theTime->tm_year + 1900; sprintf(message, "Time: %02d:%02d:%02d Date: %02d/%02d/%d\n", hours, minutes, seconds, month, day, year); - mg_send_websocket_frame(connection, WEBSOCKET_OP_TEXT, message, strlen(message)); + mg_websocket_write(connection, MG_WEBSOCKET_OPCODE_TEXT, message, strlen(message)); } int TimeSession::handleMessage(std::string client_msg) { @@ -180,14 +187,15 @@ LIBRARY DEPENDENCIES: ( (cannon/gravity/src/cannon_init.c) (cannon/gravity/src/cannon_numeric.c) - (httpMethods/TimeSession.cpp) // <--(1) + (httpMethods/handle_HTTP_GET_hello.c) + (httpMethods/TimeSession.cpp) ) *************************************************************/ #include "sim_objects/default_trick_sys.sm" -#include "sim_objects/WebServer.sm" // <--(2) +#include "sim_objects/CivetServer.sm" ##include "cannon/gravity/include/cannon_numeric.h" -##include "httpMethods/TimeSession.hh" // <--(3) +##include "httpMethods/TimeSession.hh" class CannonSimObject : public Trick::SimObject { @@ -204,11 +212,12 @@ class CannonSimObject : public Trick::SimObject { } ; CannonSimObject dyn ; -IntegLoop dyn_integloop (0.10) dyn; +IntegLoop dyn_integloop (0.01) dyn; void create_connections() { dyn_integloop.getIntegrator(Runge_Kutta_4, 5); - web.server.installWebSocketSessionMaker("Time", &makeTimeSession); // <--(4) + web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello ); + web.server.installWebSocketSessionMaker( "Time", &makeTimeSession ); } ``` diff --git a/docs/documentation/web/HTTP-API_alloc_info.html b/docs/documentation/web/HTTP-API_alloc_info.html deleted file mode 100644 index 47f133fe..00000000 --- a/docs/documentation/web/HTTP-API_alloc_info.html +++ /dev/null @@ -1,633 +0,0 @@ - - - - - - - -HTTP-API_alloc_info - - - - - - - - - - - -

HTTP-API: alloc_info

- -

http://localhost:8888/api/http/alloc_info

- -

Purpose

- -

Request a JSON encoded sub-list of allocation descriptors from the Memory Manager’s alloc_info list.

- -

Query String Parameters

- - - - - - - - - - - - - - - - - - - - - - -
ParameterDefaultDescription
start0starting index of the sub-list.
count20number of allocation descriptors.
- -

EXAMPLE:

- -

http://localhost:8888/api/http/alloc_info?start=20&count=2

- -

Query Response

- -

Returns a JSON object containing four name-value pairs:

- -

JSON Response Object

- - - - - - - - - - - - - - - - - - - - - - - - - - - -
NameValue Description
alloc_totalTotal number allocations in the Memory Manager’s alloc_info list.
chunk_sizeNumber of allocation description objects in alloc_list.
chunk_startThe Memory Manager alloc_info index of the first alloc_list element below
alloc_listArray of JSON Allocation Description objects (described below).
- -

JSON Allocation Description Object

- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
NameValue Description
nameName of the allocation. May be Null
startStarting address of the allocation.
endEnding address of the allocation.
num
sizeSize of the allocation in bytes.
typeType descriptor of the allocation.
stclStorage class of the allocation. Either TRICK_EXTERN or TRICK_LOCAL.
languageLanguage. Either : Language_C or Language_CPP.
indexArray dimension sizes of the allocation (if it represents an array).
- -

Example:

- -

In SIM_cannon_numeric (one of Trick's example sims) the following query resulted in the following JSON.

- -

Query

- -

http://localhost:8888/api/http/alloc_info?start=20&count=2

- -

Response

- -
{ "alloc_total":43,
-  "chunk_size":2,
-  "chunk_start":20,
-  "alloc_list":[
-                 { "name":"dyn",
-                   "start":"0x101aa9900",
-                   "end":"0x101aa9b27",
-                   "num":"1",
-                   "size":"552",
-                   "type":"CannonSimObject",
-                   "stcl":"TRICK_EXTERN",
-                   "language":"Language_CPP",
-                   "index": []
-                 }
-                 ,
-                 { "name":"web",
-                   "start":"0x101aa9610",
-                   "end":"0x101aa98ff",
-                   "num":"1",
-                   "size":"752",
-                   "type":"WebServerSimObject",
-                   "stcl":"TRICK_EXTERN",
-                   "language":"Language_CPP",
-                   "index": []
-                 }
-               ]
-}
-
- - - - - - - - - - - - - diff --git a/docs/documentation/web/WS-API_VariableServer.html b/docs/documentation/web/WS-API_VariableServer.html deleted file mode 100644 index c55b2320..00000000 --- a/docs/documentation/web/WS-API_VariableServer.html +++ /dev/null @@ -1,659 +0,0 @@ - - - - - - - -WS-API_VariableServer - - - - - - - - - - - -

WS-API: VariableServer

- -

ws://localhost:8888/api/ws/VariableServer

- -

Purpose

- -

JSON Variable Server

- -

Client to Server Command Messages

- -

Add a Trick Variable to the current session.

- -
{ "cmd" : "var_add",
-  "var_name" : string
-}
- -

Stop sending periodic var_list messages (see below) from the server.

- -
{ "cmd" : "var_pause" }
- -

Resume sending periodic var_list response messages from the server.

- -
{ "cmd" : "var_unpause" }
-
- -

Send one var_list message from the server.

- -
{ "cmd" : "var_send" }
- -

Clear all variables from the current session, that is: undo all of the var_add commands.

- -
{ "cmd" : "var_clear" }
- -

Disconnect from the variable server.

- -
{ "cmd" : "var_exit" }
- -

Set the period (in milliseconds) at which var_list messages are sent form the server.

- -
{ "cmd" : "var_cycle",
-  "period" : integer
-}
- -

Execute the given Python code in the host sim.

- -
{ "cmd" : "python",
-  "pycode" : string
-}
- -

Send the sie structure from the server. Response will be the sie response message (below).

- -
{ "cmd" : "sie" }
- -

Send the units for the given variable. Response will be the units response message (below).

- -
{ "cmd" : "units",
-  "var_name" : string
-}
- -

Server to Client Response Messages

- -

Error Response

- -
{ "msg_type" : "error",
-  "error_text" : string
-}
- -

Periodic response containing the values of variables requested by var_add.

- -
{ "msg_type" : "var_list"
-  "time" : double
-  "values" : []
-}
- -

Response to the sie command (above).

- -
{ "msg_type" : "sie",
-  "data" : string
-}
- -

Response to the units command (above).

- -
{ "msg_type" : "units",
-  "var_name" : string,
-  "data" : string
-}
- -

Example Variable Server Client

- -
<!DOCTYPE html>
-<html>
-  <head>
-    <title>WS Experiments</title>
-  </head>
-  <body>
-      <style>
-          table { border-collapse: collapse; width: 100%; }
-          th, td { text-align: left; padding: 8px; }
-          tr:nth-child(even){background-color: #f2f2f2}
-          th { background-color: #562399; color: white; }
-      </style>
-      <header>
-      </header>
-
-      <div class="variableDisplay"></div>
-      <table class="variables">
-          <tr>
-              <th>Variable</th>
-              <th>Value</th>
-          </tr>
-      </table>
-
-      <div id="output"></div>
-      <script type="text/javascript">
-          function log(s) {
-              var p = document.createElement("p");
-              p.style.wordWrap = "break-word";
-              p.textContent = s;
-              output.appendChild(p);
-          }
-          function sendMessage(msg) {
-              ws.send(msg);
-          }
-          // Interface to Trick WebSocket Variable Server
-          function setPeriod(period) {
-              sendMessage(`{"cmd":"var_cycle","period":${period}}`);
-          }
-          function addVarTableRow(name, value) {
-              // create a row in the table that contains two <td>s, one for the var_name and one for its value.
-              let tr = document.createElement('tr');
-              let td1 = document.createElement('td');
-              td1.textContent = `${name}`;
-              let td2 = document.createElement('td');
-              td2.textContent = `${value}`;
-              td2.className = "values";
-              tr.appendChild(td1);
-              tr.appendChild(td2);
-              varTable.appendChild(tr);
-          }
-          function addVariable(name, value) {
-              sendMessage(`{"cmd":"var_add","var_name": "${name}"}`);
-              addVarTableRow(name, value);
-          }
-          var varTable = document.querySelector('table.variables');
-
-
-          var ws = new WebSocket('ws://localhost:8888/api/ws/VariableServer');
-          ws.onopen = function(e) {
-              setPeriod(100);
-              addVarTableRow("Time", 0.0);
-              addVariable("dyn.cannon.pos[0]", 0.0);
-              addVariable("dyn.cannon.pos[1]", 0.0);
-              addVariable("dyn.cannon.vel[0]", 0.0);
-              addVariable("dyn.cannon.vel[1]", 0.0);
-              addVariable("dyn.cannon.time", 0.0);
-              addVariable("dyn.cannon.timeRate", 0.0);
-              addVariable("dyn.cannon.impact", 0.0);
-              addVariable("I.dont.exist", 0.0);
-              sendMessage("{\"cmd\":\"var_unpause\"}");
-          };
-          ws.onmessage = function(e) {
-             let msg = JSON.parse(e.data);
-             if (msg.msg_type == "values") {
-                 let valueNodes = varTable.getElementsByClassName("values");
-                 valueNodes[0].textContent = msg.time;
-                 for (let i = 0; i < msg.values.length; i++ ) {
-                     valueNodes[i+1].textContent = msg.values[i];
-                 }
-             }
-          };
-          ws.onerror = function(e) {
-             console.log("WebSocket Error: " , e);
-             handleErrors(e);
-          };
-          ws.onclose = function(e) {
-             console.log("Connection closed", e);
-          };
-
-      </script>
-  </body>
-</html>
- - - - - - - - - - - - - - - diff --git a/docs/documentation/web/index.html b/docs/documentation/web/index.html deleted file mode 100644 index 85547ad3..00000000 --- a/docs/documentation/web/index.html +++ /dev/null @@ -1,44 +0,0 @@ - - - - - - Web Server Documentation -
- - - -

Web Server Documentation

-
- - - - -
-
- - - diff --git a/include/trick/MyCivetServer.hh b/include/trick/MyCivetServer.hh new file mode 100644 index 00000000..086f265d --- /dev/null +++ b/include/trick/MyCivetServer.hh @@ -0,0 +1,85 @@ +/************************************************************************* +PURPOSE: (Represent the state and initial conditions of an http server.) +**************************************************************************/ +#ifndef CIVET_SERVER_H +#define CIVET_SERVER_H + +#ifdef USE_CIVET + +#include +#include +#include +#include +#include +#include +#include "trick/WebSocketSession.hh" + +typedef WebSocketSession* (*WebSocketSessionMaker)(struct mg_connection *nc); +typedef void (*httpMethodHandler)(struct mg_connection *, void* cbdata); + +class MyCivetServer { + public: + + unsigned int port; + const char* document_root; + bool enable; + bool debug; + bool time_homogeneous; + const char* path_to_ssl_cert; + bool ssl_enable; + const char* error_log_file; + + struct mg_context *ctx; /* ** civetweb */ + + // Trick Job-Functins + int default_data(); + int shutdown(); + int init(); + int join(); + int http_top_of_frame(); + + //TODO: Make these private and fix threading design issue + pthread_t server_thread; /* ** */ + bool sessionDataMarshalled; /* ** */ + pthread_mutex_t lock_loop; /* ** */ + pthread_cond_t cond_loop; /* ** */ + bool service_connections = true; /* ** */ + bool shutting_down; /* ** */ + + std::map WebSocketSessionMakerMap; /* ** */ + pthread_mutex_t WebSocketSessionMakerMapLock; /* ** */ + + std::map webSocketSessionMap; /* ** */ + pthread_mutex_t WebSocketSessionMapLock; /* ** */ + + std::map< std::string, httpMethodHandler> httpGETHandlerMap; /* ** */ + pthread_mutex_t httpGETHandlerMapLock; /* ** */ + + void addWebSocketSession(struct mg_connection *nc, WebSocketSession* session); + WebSocketSession* makeWebSocketSession(struct mg_connection *nc, std::string name); + void marshallWebSocketSessionData(); + void sendWebSocketSessionMessages(struct mg_connection *nc); + void unlockConnections(); + void deleteWebSocketSession(struct mg_connection * nc); + void installHTTPGEThandler(std::string handlerName, httpMethodHandler handler); + void installWebSocketSessionMaker(std::string name, WebSocketSessionMaker maker); + void handleWebSocketClientMessage(struct mg_connection *conn, const char* data); + void handleHTTPGETrequest(struct mg_connection *conn, const struct mg_request_info* ri, std::string handlerName); + + + std::string tmp_string; + + + + // void installWebSocketSessionMaker(std::string name, WebSocketSessionMaker maker); + + +}; + +struct Data { + MyCivetServer* server; + std::string name; +}; + +#endif +#endif diff --git a/include/trick/WebServer.hh b/include/trick/WebServer.hh deleted file mode 100644 index 14688231..00000000 --- a/include/trick/WebServer.hh +++ /dev/null @@ -1,69 +0,0 @@ -/************************************************************************* -PURPOSE: (Represent the state and initial conditions of an http server.) -**************************************************************************/ -#ifndef WEB_SERVER_H -#define WEB_SERVER_H - -#ifdef USE_MONGOOSE - -#include -#include -#include -#ifndef SWIG -#include "mongoose/mongoose.h" -#endif -#include "trick/WebSocketSession.hh" - -typedef void (*httpMethodHandler)(struct mg_connection *, struct http_message *); -typedef WebSocketSession* (*WebSocketSessionMaker)(struct mg_connection *nc); - -class WebServer { - public: - const char* port; - const char* document_root; - struct mg_mgr mgr; /* ** mongoose */ - struct mg_connection *listener; /* ** mongoose */ - pthread_t server_thread; /* ** */ - bool shutting_down; - bool enable; - bool debug; - - std::map< std::string, httpMethodHandler> httpGETHandlerMap; /* ** */ - pthread_mutex_t httpGETHandlerMapLock; /* ** */ - - std::map< std::string, WebSocketSessionMaker> WebSocketSessionMakerMap; /* ** */ - pthread_mutex_t WebSocketSessionMakerMapLock; /* ** */ - - std::map webSocketSessionMap; /* ** */ - pthread_mutex_t webSocketSessionMapLock; /* ** */ - - pthread_mutex_t serviceLock; /* ** */ - struct mg_serve_http_opts http_server_options; /* ** */ - struct mg_bind_opts bind_opts; /* ** */ - pthread_cond_t serviceConnections; /* ** */ - bool service_websocket; - bool time_homogeneous; - bool sessionDataMarshalled; - - // Trick Job-functions - int http_default_data(); - int http_init(); - int http_top_of_frame(); - int http_shutdown(); - - void installWebSocketSessionMaker(std::string name, WebSocketSessionMaker maker); - void installHTTPGEThandler(std::string handlerName, httpMethodHandler handler); - - // These are internals, and should not be considered public. They are not private only - // because they need to be callable from the servers event handler. - void sendWebSocketSessionMessages(struct mg_connection *nc); - void handleWebSocketClientMessage(struct mg_connection *nc, std::string msg); - void addWebSocketSession(struct mg_connection *nc, WebSocketSession* session); - void deleteWebSocketSession(struct mg_connection *nc); - WebSocketSession* makeWebSocketSession(struct mg_connection *nc, std::string name); - void handleHTTPGETrequest(struct mg_connection *nc, http_message *hm, std::string handlerName); - void marshallWebSocketSessionData(); -}; - -#endif -#endif diff --git a/include/trick/WebSocketSession.hh b/include/trick/WebSocketSession.hh index 389e33f5..a7f6ea80 100644 --- a/include/trick/WebSocketSession.hh +++ b/include/trick/WebSocketSession.hh @@ -4,10 +4,10 @@ PURPOSE: (Represent Websocket connection.) #ifndef WEB_SOCKET_SESSION_HH #define WEB_SOCKET_SESSION_HH -#ifdef USE_MONGOOSE +#ifdef USE_CIVET #include #ifndef SWIG -#include "mongoose/mongoose.h" +#include "civet/CivetServer.h" #endif class WebSocketSession { @@ -28,4 +28,4 @@ class WebSocketSession { }; #endif -#endif +#endif \ No newline at end of file diff --git a/include/trick/files_to_ICG.hh b/include/trick/files_to_ICG.hh index 161f8fb1..fde05cd6 100644 --- a/include/trick/files_to_ICG.hh +++ b/include/trick/files_to_ICG.hh @@ -97,8 +97,8 @@ #include "trick/lqueue.h" #include "trick/lstack.h" -#ifdef USE_MONGOOSE -#include "trick/WebServer.hh" +#ifdef USE_CIVET +#include "trick/MyCivetServer.hh" #include "trick/WebSocketSession.hh" #endif diff --git a/share/trick/makefiles/Makefile.common b/share/trick/makefiles/Makefile.common index 43d60124..0bf65c20 100644 --- a/share/trick/makefiles/Makefile.common +++ b/share/trick/makefiles/Makefile.common @@ -171,11 +171,13 @@ ifneq ($(GSL_HOME),) TRICK_SYSTEM_CXXFLAGS += -D_HAVE_GSL endif -ifeq (${TRICK_MONGOOSE},1) - TRICK_LIBS += -ltrickHTTP ${TRICK_LIB_DIR}/libmongoose.a - TRICK_SWIG_FLAGS += -DUSE_MONGOOSE - TRICK_SYSTEM_CXXFLAGS += -DUSE_MONGOOSE - TRICK_ICG_EXCLUDE += :${TRICK_HOME}/include/mongoose +TRICK_CIVET=0 +ifneq (${TRICK_DISABLE_CIVET},1) + TRICK_CIVET=1 + TRICK_LIBS += ${TRICK_LIB_DIR}/libcivetweb.a ${TRICK_LIB_DIR}/libtrickCivet.a + TRICK_SWIG_FLAGS += -DUSE_CIVET + TRICK_SYSTEM_CXXFLAGS += -DUSE_CIVET + TRICK_ICG_EXCLUDE += ${TRICK_HOME}/include/civet endif # We pipe the output of compiler through tee. If the user wanted gcc color, make sure they get it. diff --git a/share/trick/makefiles/config_user.mk.in b/share/trick/makefiles/config_user.mk.in index 2fe8610b..52ffedda 100644 --- a/share/trick/makefiles/config_user.mk.in +++ b/share/trick/makefiles/config_user.mk.in @@ -17,7 +17,7 @@ USE_JAVA = @USE_JAVA@ JAVAC = @JAVA_CC@ TRICK_OFFLINE = @TRICK_OFFLINE@ -TRICK_MONGOOSE = @TRICK_MONGOOSE@ +TRICK_DISABLE_CIVET = @TRICK_DISABLE_CIVET@ USE_X_WINDOWS = @USE_X_WINDOWS@ diff --git a/share/trick/pymods/trick/.gitignore b/share/trick/pymods/trick/.gitignore new file mode 100644 index 00000000..1c39d874 --- /dev/null +++ b/share/trick/pymods/trick/.gitignore @@ -0,0 +1,6 @@ +*venv +.vscode/* +*.pem +*.key +*.crt +*.csr diff --git a/share/trick/pymods/trick/conftest.py b/share/trick/pymods/trick/conftest.py new file mode 100644 index 00000000..b3961b6c --- /dev/null +++ b/share/trick/pymods/trick/conftest.py @@ -0,0 +1,68 @@ +import pytest +import sys +import os +from typing import Dict, Tuple +import subprocess +import inspect + +sys.path.append(os.path.abspath(os.path.join(os.path.dirname(os.path.abspath(inspect.getsourcefile(lambda:0))), '../..'))) +from utils import is_web_server_started, params, pause + +# store history of failures per test class name and per index in parametrize (if parametrize used) +web_server_status = {} +web_server_status2 = None + +def build_sim(): + with open(os.path.join(params.get_path_to_sim(), params.get_input_folder(), params.get_test_input_file()), "w") as f: + f.write( \ + f"""web.server.enable = True +web.server.debug = False +web.server.ssl_enable = {params.get_ssl_enable()} +web.server.path_to_ssl_cert = '{params.get_ssl_cert_path()}' +web.server.port = {params.get_port()} + +trick.var_server_set_port({params.get_var_server_port()}) + +trick.frame_log_on() +trick.real_time_enable() +trick.exec_set_software_frame(0.1) +trick.itimer_enable() + +trick.exec_set_enable_freeze(True) +trick.exec_set_freeze_command(True)""") + + pathToSim=params.get_path_to_sim() + if params.get_build_sim(): + #TODO: Need make file to only rebuild only when necessary, otherwise, test need to rebuild and this is time consuming. + print("#"*10) + print("Auto rebuilding sim. Auto rebuild will build the SIM everytime the test is run, which can take some time.") + print("To turn auto rebuild off, in utils.py, self.__build_sim = False. Note: it's important that SIM rebuild is current.") + print("#"*10) + + build_cmd = f"echo \"cd {pathToSim} && make -C {params.get_trick_home()}/trick_source/web/CivetServer\" | /bin/bash" #TODO: Only rebuild if necessary. + print("....................Running:", build_cmd) + subprocess.run(build_cmd, shell=True) + + print("Directory listing:") + os.listdir(".") + + build_cmd = f"echo \"cd {pathToSim} && {params.get_trick_home()}/bin/trick-CP\" | /bin/bash" #TODO: perform a make clean if webserver code gets updated because trick-CP will no update the sim to include in changes to the webserver. + print("....................Running:", build_cmd) + subprocess.run(build_cmd, shell=True) + + if params.get_start_sim(): + if not os.path.exists(os.path.join(pathToSim, params.get_sim_name())): + raise RuntimeError(f"Sim executable does not exist in {pathToSim}. Build this sim before running this test.") + cmd = f'echo "cd {pathToSim} && ./{params.get_sim_name()} {os.path.join(params.get_input_folder(), params.get_test_input_file())} &" | /bin/bash' + print("....................Running:", cmd) + subprocess.run(cmd, shell=True) + +@pytest.fixture(scope="session", autouse=True) +def close_sim(): + build_sim() + if not is_web_server_started(): + pytest.fail("web server is not started.") + yield + if params.get_start_sim(): + os.system("pgrep S_ | xargs kill -9") + os.remove(os.path.join(params.get_path_to_sim(), params.get_input_folder(), params.get_test_input_file())) diff --git a/share/trick/pymods/trick/makefile b/share/trick/pymods/trick/makefile new file mode 100644 index 00000000..eb612917 --- /dev/null +++ b/share/trick/pymods/trick/makefile @@ -0,0 +1,12 @@ +PYTHON = python + +test: + $(PYTHON) -m pytest tests/civet_server/test_http.py tests/civet_server/test_ws.py # tests/gsetup + +venv: + $(PYTHON) -m pip install virtualenv + $(PYTHON) -m virtualenv venv + venv/bin/python -m pip install -r requirements.txt + +clean: + rm -rf venv \ No newline at end of file diff --git a/share/trick/pymods/trick/pytest.ini b/share/trick/pymods/trick/pytest.ini new file mode 100644 index 00000000..814f6b57 --- /dev/null +++ b/share/trick/pymods/trick/pytest.ini @@ -0,0 +1,3 @@ +[pytest] +markers = + webserver: Tests relies on the webserver \ No newline at end of file diff --git a/share/trick/pymods/trick/requirements.txt b/share/trick/pymods/trick/requirements.txt new file mode 100644 index 00000000..0cfe2244 --- /dev/null +++ b/share/trick/pymods/trick/requirements.txt @@ -0,0 +1,5 @@ +websockets +pytest-asyncio +pytest +requests +psutil diff --git a/share/trick/pymods/trick/run_tests.py b/share/trick/pymods/trick/run_tests.py new file mode 100755 index 00000000..2aaec1c5 --- /dev/null +++ b/share/trick/pymods/trick/run_tests.py @@ -0,0 +1,8 @@ +#!/usr/bin/env python3 +import pytest +import sys + +if __name__ == "__main__": + args = list(sys.argv[1:]) + ["tests/civet_server"] + print("Runnig tests with arguments:", args) + sys.exit(pytest.main(args)) \ No newline at end of file diff --git a/share/trick/pymods/trick/tests/civet_server/test_http.py b/share/trick/pymods/trick/tests/civet_server/test_http.py new file mode 100644 index 00000000..af5cb9db --- /dev/null +++ b/share/trick/pymods/trick/tests/civet_server/test_http.py @@ -0,0 +1,107 @@ +from sys import path +import pytest +import requests +from pprint import pprint +import logging +import os +import subprocess +from time import sleep +import shutil +import datetime +import socket + +path.append(os.path.join(os.environ.get("TRICK_HOME", "../../../.."), "trick_source/trick_gsetup")) +from requests.api import get, request + +# TODO: Get rid of this and use automatic discovery when Trick requires Python 2.7 +path.append("../..") +from utils import is_web_server_started, params + +def open_connections(numConnections): + sockets = [] + for _ in range(numConnections): + sockets.append(socket.socket(socket.AF_INET, socket.SOCK_STREAM)) + sockets[-1].connect(("127.0.0.1", params.get_var_server_port())) + sleep(1) #Wait for the connection to persist. + return sockets + +def close_sockets(sockets): + for s in sockets: + port = s.getsockname()[1] + s.close() + # is_web_server_started(port, "") + sleep(1) + +def get_vs_open_connections(): + endpoint = "api/http/vs_connections" + url = params.get_url(endpoint) + res = requests.get(url, verify=False) + print("Array:",res.json()["variable_server_connections"]) + return res + +@pytest.mark.webserver +class TestWebserverHttp: + def test_404_URL_not_found(self): + url = params.get_url("api/http/invalid") + res = requests.get(url, verify=False) #verify=False says to not verify https for self-signed cert + assert res.status_code == 404, f"Requested URL should not exist. Status code 404 was not returned. Response text:\n#######\n{res.text}\n#######" + + def test_alloc_info(self): + url = params.get_url("api/http/alloc_info") + res = requests.get(url, verify=False) #verify=False says to not verify https for self-signed cert + print(f"Response: {res.text}") + assert res.status_code == 200, "Requested URL does not exist." + data = res.json() + assert len(data["alloc_list"]) == 10, "Expecting default &count to be 10." + assert data["chunk_size"] == 10, "Expecting default &count to be 10." + assert data["chunk_start"] == 0, "expecting default &start to be 0." + assert data["alloc_total"] == 48, "Expecting 48 memory allocations." + + def test_alloc_info_2(self): + start = 2 + count = 12 + endpoint = f"api/http/alloc_info?start={start}&count={count}" + url = params.get_url(endpoint) + res = requests.get(url, verify=False) + assert len(res.json()["alloc_list"]) == count + assert res.json()["chunk_start"] == start + assert res.json()["alloc_total"] == 48, "Expecting 48 memory allocations." + + def test_vs_connections(self): + sockets = open_connections(1) + webResponse=get_vs_open_connections() + close_sockets(sockets) + assert webResponse.json()["variable_server_connections"][0]["connection"]["client_IP_address"] == "127.0.0.1" + assert len(webResponse.json()["variable_server_connections"]) == 1 + + sockets = open_connections(50) + webResponse=get_vs_open_connections() + close_sockets(sockets) + assert webResponse.json()["variable_server_connections"][0]["connection"]["client_IP_address"] == "127.0.0.1" + assert len(webResponse.json()["variable_server_connections"]) == 50, "Should be able to open more than 1 connection." #Todo: determine appropriate number of simultaneous connections to test + + assert len(get_vs_open_connections().json()["variable_server_connections"]) == 0 + + def test_index(self): + url = params.get_url("index.html") + res = requests.get(url) + assert res.status_code == 200, "No index.html file served" + + def test_post(self): + url = params.get_url("api/http/alloc_info") + res = requests.post(url) + assert res.status_code == 405, "Did not receive a method not allowed 405 error" + + def test_delete(self): + url = params.get_url("api/http/alloc_info") + res = requests.delete(url) + assert res.status_code == 405, "Did not receive a method not allowed 405 error" + + def test_http_headers(self): + url = params.get_url("") + res = requests.get(url) + assert res.headers["Content-Type"] == 'text/html' + + +if __name__ == "__main__": + pass \ No newline at end of file diff --git a/share/trick/pymods/trick/tests/civet_server/test_misc.py b/share/trick/pymods/trick/tests/civet_server/test_misc.py new file mode 100644 index 00000000..ec2dd636 --- /dev/null +++ b/share/trick/pymods/trick/tests/civet_server/test_misc.py @@ -0,0 +1,9 @@ +import websockets +import pytest +import sys + +# sys.path.append("../..") +# from parameters import Params +# from test_ws import ssl_context + + diff --git a/share/trick/pymods/trick/tests/civet_server/test_ws.py b/share/trick/pymods/trick/tests/civet_server/test_ws.py new file mode 100644 index 00000000..fb87809a --- /dev/null +++ b/share/trick/pymods/trick/tests/civet_server/test_ws.py @@ -0,0 +1,100 @@ +import logging +import json +import pytest +import websockets +import asyncio +from time import sleep +import datetime +import sys +import os +import pathlib +import ssl +import platform + +sys.path.append("../..") +from utils import is_web_server_started, params + +@pytest.mark.webserver +class TestWebserverWs: + if params.get_ssl_enable(): + ssl_context = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT) + # localhost_pem = pathlib.Path(__file__).with_name(params.get_ssl_cert_path()) + localhost_pem = params.get_ssl_cert_path() + ssl_context.load_verify_locations(localhost_pem) + else: + ssl_context = None + + @pytest.fixture(autouse=True, scope="session") + def variable_server_path(self): + return params.get_ws_url("api/ws/VariableServer") + + @pytest.fixture(autouse=True, scope="session") + def time_path(self): + return params.get_ws_url("api/ws/Time") + + @pytest.mark.asyncio + async def test_time(self, time_path): + if params.get_test_time(): + async with websockets.connect(time_path, ssl=TestWebserverWs.ssl_context) as websocket: + await websocket.send("LOCAL") + count = 0 + while count < 2: + message = await websocket.recv() + test_format = "Time: %H:%M Date: %m/%d/%Y\n" #Not checking seconds. + time = datetime.datetime.strftime(datetime.datetime.strptime(message, "Time: %H:%M:%S Date: %m/%d/%Y\n"), test_format) + test_time = datetime.datetime.now().strftime(test_format) + print("Checking:", time, "=", test_time) + assert time == test_time + count += 1 + else: + raise RuntimeError("Parameter test_time is disabled.") + + @pytest.mark.asyncio + async def test_variable_server_vars(self, variable_server_path): + msg1 = '{"cmd":"var_add","var_name":"dyn.cannon.pos[0]"}' + msg2 = '{ "cmd" : "var_send" }' + async with websockets.connect(variable_server_path, ssl=TestWebserverWs.ssl_context) as websocket: + await websocket.send(msg1) + await websocket.send(msg2) + message = await websocket.recv() + vars = json.loads(message) + assert vars["msg_type"] == "values" + assert "time" in vars + assert len(vars["values"]) == 1 + + @pytest.mark.asyncio + async def test_variable_server_sie(self, variable_server_path): + async with websockets.connect(variable_server_path, ssl=TestWebserverWs.ssl_context) as websocket: + await websocket.send('{ "cmd" : "sie" }') + response = await websocket.recv() + assert response == '{ "msg_type": "sie", "data": ' + + @pytest.mark.asyncio + async def test_variable_server_units(self, variable_server_path): + msg1 = '{"cmd":"var_add","var_name":"dyn.cannon.pos[0]"}' + msg2 = '{ "cmd" : "units", "var_name" : "dyn.cannon.pos[0]" }' + async with websockets.connect(variable_server_path, ssl=TestWebserverWs.ssl_context) as websocket: + await websocket.send(msg1) + await websocket.send(msg2) + response = await websocket.recv() + assert response == '{ "msg_type": "units", "var_name": "dyn.cannon.pos[0]", "data": "m"}' + + @pytest.mark.asyncio + async def test_variable_server_shell_access(self, variable_server_path): + async with websockets.connect(variable_server_path, ssl=TestWebserverWs.ssl_context) as websocket: + file_to_create = "tmp_a.txt" + await websocket.send('{ "cmd" : "python", "pycode" : "print \'Hello World!---------------------\'" }') + await websocket.send('{ "cmd" : "python", "pycode" : "import os" }') + await websocket.send('{ "cmd" : "python", "pycode" : "os.system(\'touch ' + file_to_create + '\')" }') + path = os.path.join(params.get_path_to_sim(), file_to_create) + if os.path.exists(path): + os.remove(path) + warning = "This test proves that we have shell access through the websocket api. Is this a security concern? Should this test fail if shell access is available?" + print(warning) + assert 1 + # raise RuntimeError(warning) + else: + assert 0 + +if __name__ == "__main__": + pass \ No newline at end of file diff --git a/share/trick/pymods/trick/utils.py b/share/trick/pymods/trick/utils.py new file mode 100644 index 00000000..a18463cf --- /dev/null +++ b/share/trick/pymods/trick/utils.py @@ -0,0 +1,100 @@ +from time import sleep +import subprocess +import os +import psutil + +def pause(my_str = "no message."): + print("Type exit to continue:" + my_str) + # os.system("/bin/bash") + +#This file contains variables for the civet_server tests +class Params: + #Change the following to change the default parameters + def __init__(self) -> None: + self.__port = 9000 + self.__var_server_port = 9001 + self.__host = "localhost" + self.__enable_ssl = False + self.__test_time = True + self.__ssl_cert_path = os.path.join(os.environ["TRICK_HOME"], "trick_sims", "Cannon", "SIM_cannon_webserver", "server.pem") + self.__build_sim = True + self.__start_sim = True + self.__trick_home = os.environ["TRICK_HOME"] + if not self.__trick_home: + print("ERROR:", "TRICK_HOME not found in environment variables.") + self.__input_folder = "RUN_test" + self.__test_input_file = f"tmp_input_for_test.py" + + def get_sim_name(self): + sim_name = None + for file in os.listdir(self.get_path_to_sim()): + if file.startswith("S_main"): + sim_name = file + if sim_name == None: + raise RuntimeError(f"Did not find sim executable. Please make sure the sim in {self.get_path_to_sim()} is compiled.") + return sim_name + def get_trick_home(self): + return self.__trick_home + def get_path_to_sim(self): + return os.path.join(self.get_trick_home(), "trick_sims", "Cannon", "SIM_cannon_webserver") + def get_input_folder(self): + return self.__input_folder + def get_test_input_file(self): + return self.__test_input_file + + def get_start_sim(self): + return self.__start_sim + + def get_build_sim(self): + return self.__build_sim + + def get_ssl_cert_path(self): + return self.__ssl_cert_path + + def get_port(self): + return self.__port + + def get_host(self): + if self.get_ssl_enable(): + return self.__host + ".ssl" + else: + return self.__host + + def get_ssl_enable(self): + return self.__enable_ssl + + def get_var_server_port(self): + return self.__var_server_port + + def get_test_time(self): + return self.__test_time + + def get_url(self, endpoint): + server_port = self.get_port() + server_host = self.get_host() + ssl_enable = self.get_ssl_enable() + base_url = f"http{ 's' if ssl_enable else '' }://{server_host}:{server_port}" + return f"{base_url}/{endpoint}" + + def get_ws_url(self, endpoint): + return f"ws{ 's' if self.get_ssl_enable() else '' }://{self.get_host()}:{self.get_port()}/{endpoint}" + +params = Params() + +def is_web_server_started(port=params.get_port(), status_method="LISTEN"): + isConnectionOpen = False + try: + for _ in range(20): #Wait up to 2 seconds i.e 20 * .1 seconds, must wait for service to get to listening state. + for connection in psutil.net_connections(): + local_address = connection.laddr + if len(local_address) > 1 and local_address[1] == port and connection.status == status_method: + isConnectionOpen = True + break + if isConnectionOpen: + break + sleep(.1) #We sleep to use less recourses + except psutil.AccessDenied as e: + print("psutil.net_connections() requires root access on mac. Sleeping for 2 seconds instead.") + isConnectionOpen = True + sleep(2) + return isConnectionOpen diff --git a/share/trick/sim_objects/CivetServer.sm b/share/trick/sim_objects/CivetServer.sm new file mode 100644 index 00000000..48892acd --- /dev/null +++ b/share/trick/sim_objects/CivetServer.sm @@ -0,0 +1,20 @@ +/************************TRICK HEADER************************* +PURPOSE: (Trick HTTP Server) +*************************************************************/ +##include "trick/MyCivetServer.hh" + +class MyCivetServerSimObject : public Trick::SimObject { + + public: + MyCivetServer server ; + + MyCivetServerSimObject() { + ("default_data") server.default_data() ; + ("initialization") server.init() ; + ("freeze") server.http_top_of_frame() ; + ("top_of_frame") server.http_top_of_frame() ; + ("shutdown") server.shutdown() ; + } +}; + +MyCivetServerSimObject web; diff --git a/share/trick/sim_objects/WebServer.sm b/share/trick/sim_objects/WebServer.sm deleted file mode 100644 index 66774b5f..00000000 --- a/share/trick/sim_objects/WebServer.sm +++ /dev/null @@ -1,20 +0,0 @@ -/************************TRICK HEADER************************* -PURPOSE: (Trick HTTP Server) -*************************************************************/ -##include "trick/WebServer.hh" - -class WebServerSimObject : public Trick::SimObject { - - public: - WebServer server ; - - WebServerSimObject() { - ("default_data") server.http_default_data() ; - ("initialization") server.http_init() ; - ("freeze") server.http_top_of_frame() ; - ("top_of_frame") server.http_top_of_frame() ; - ("shutdown") server.http_shutdown() ; - } -}; - -WebServerSimObject web; diff --git a/trick_sims/Cannon/SIM_cannon_numeric/S_define b/trick_sims/Cannon/SIM_cannon_numeric/S_define index 48b554c0..ef7d6480 100644 --- a/trick_sims/Cannon/SIM_cannon_numeric/S_define +++ b/trick_sims/Cannon/SIM_cannon_numeric/S_define @@ -9,7 +9,7 @@ LIBRARY DEPENDENCIES: *************************************************************/ #include "sim_objects/default_trick_sys.sm" -// #include "sim_objects/WebServer.sm" +//#include "sim_objects/CivetServer.sm" ##include "cannon/gravity/include/cannon_numeric.h" class CannonSimObject : public Trick::SimObject { diff --git a/trick_sims/Cannon/SIM_cannon_webserver/Modified_data/cannon.dr b/trick_sims/Cannon/SIM_cannon_webserver/Modified_data/cannon.dr new file mode 100644 index 00000000..8810bedd --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/Modified_data/cannon.dr @@ -0,0 +1,17 @@ +global DR_GROUP_ID +global drg +try: + if DR_GROUP_ID >= 0: + DR_GROUP_ID += 1 +except NameError: + DR_GROUP_ID = 0 + drg = [] + +drg.append(trick.DRAscii("cannon")) +drg[DR_GROUP_ID].set_freq(trick.DR_Always) +drg[DR_GROUP_ID].set_cycle(0.01) +drg[DR_GROUP_ID].set_single_prec_only(False) +drg[DR_GROUP_ID].add_variable("dyn.cannon.pos[0]") +drg[DR_GROUP_ID].add_variable("dyn.cannon.pos[1]") +trick.add_data_record_group(drg[DR_GROUP_ID], trick.DR_Buffer) +drg[DR_GROUP_ID].enable() diff --git a/trick_sims/Cannon/SIM_cannon_webserver/Modified_data/realtime.py b/trick_sims/Cannon/SIM_cannon_webserver/Modified_data/realtime.py new file mode 100644 index 00000000..9998f5b6 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/Modified_data/realtime.py @@ -0,0 +1,11 @@ + +trick.frame_log_on() +trick.real_time_enable() +trick.exec_set_software_frame(0.1) +trick.itimer_enable() + +#trick.exec_set_enable_freeze(True) +trick.exec_set_freeze_command(True) + +#simControlPanel = trick.SimControlPanel() +#trick.add_external_application(simControlPanel) diff --git a/trick_sims/Cannon/SIM_cannon_webserver/README.md b/trick_sims/Cannon/SIM_cannon_webserver/README.md new file mode 100644 index 00000000..a55da54e --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/README.md @@ -0,0 +1,31 @@ +# SIM\_cannon\_numeric + +--- + +This is the second in a series example cannon ball simulations that one builds in the Trick Tutorial (Sections 9, 10). It's purpose is to introduce Trick supported numerical methods. + +Rather than using the analytic solutions of SIM\_cannon\_analytic, this simulation uses Trick's numerical integration, and root-finding methods to: + +* Determine the cannon ball state (velocity, position) at each time step, and +* Determine when and where the cannon ball impacts the ground. + +The simulation is otherwise the same as SIM\_cannon\_analytic. + +![CannonPicture](images/CannonInit.png) + +## Parameterization +Same as in SIM\_cannon\_analytic. + +## Initialization +Same as in SIM\_cannon\_analytic. + +## State Propagation + +This Trick simulation calculates the cannonball state by numerical integration. +Acceleration is calculated in the "derivative" job **cannon\_deriv**. It is then +used in "integration" job **cannon\_integ** to calculate the cannonball's velocity and +position, using the Trick **integrate** function. + +## When and Where the Ball Impacts the Ground +The time and position of impact is determined by the "dynamic\_event" event job +**cannon\_impact**, using the Trick **regula_falsi** function. \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/RUN_graphics/input.py b/trick_sims/Cannon/SIM_cannon_webserver/RUN_graphics/input.py new file mode 100644 index 00000000..4cfafb49 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/RUN_graphics/input.py @@ -0,0 +1,27 @@ +exec(open("Modified_data/realtime.py").read()) + +#========================================== +# Start the Cannonball Graphics Client +#========================================== +web.server.enable = True +# web.server.debug = True +# web.server.port = 8888 +# web.server.document_root = "www" + +trick.var_server_set_port(5001); +varServerPort = trick.var_server_get_port(); +CannonDisplay_path = "../models/graphics/dist/CannonDisplay.jar" + +print("pos: ", dyn.cannon.pos[0]) + +if (os.path.isfile(CannonDisplay_path)) : + CannonDisplay_cmd = "java -jar " \ + + CannonDisplay_path \ + + " " + str(varServerPort) + " &" ; + print(CannonDisplay_cmd) + os.system( CannonDisplay_cmd); +else : + print('==================================================================================') + print('CannonDisplay needs to be built. Please \"cd\" into ../models/graphics and type \"make\".') + print('==================================================================================') + diff --git a/trick_sims/Cannon/SIM_cannon_webserver/RUN_test/input.py b/trick_sims/Cannon/SIM_cannon_webserver/RUN_test/input.py new file mode 100644 index 00000000..3af8c1d8 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/RUN_test/input.py @@ -0,0 +1,11 @@ +web.server.enable = True +web.server.debug = True +web.server.ssl_enable = False +web.server.path_to_ssl_cert = "server.pem" +web.server.error_log_file = "log.error" + +web.server.port = 8888 +trick.var_server_set_port(9001) + +exec(open("Modified_data/realtime.py").read()) +exec(open("Modified_data/cannon.dr").read()) diff --git a/trick_sims/Cannon/SIM_cannon_webserver/RUN_test/unit_test.py b/trick_sims/Cannon/SIM_cannon_webserver/RUN_test/unit_test.py new file mode 100644 index 00000000..9db27413 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/RUN_test/unit_test.py @@ -0,0 +1,3 @@ + +dyn_integloop.getIntegrator(trick.Runge_Kutta_4, 5) +trick.exec_set_terminate_time(5.2) diff --git a/trick_sims/Cannon/SIM_cannon_webserver/S_define b/trick_sims/Cannon/SIM_cannon_webserver/S_define new file mode 100644 index 00000000..4656454f --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/S_define @@ -0,0 +1,40 @@ +/***********************TRICK HEADER************************* +PURPOSE: + (Cannon Numeric) +LIBRARY DEPENDENCIES: + ( + (cannon/gravity/src/cannon_init.c) + (cannon/gravity/src/cannon_numeric.c) + (httpMethods/handle_HTTP_GET_hello.c) + (httpMethods/TimeSession.cpp) + ) +*************************************************************/ + +#include "sim_objects/default_trick_sys.sm" +#include "sim_objects/CivetServer.sm" +##include "cannon/gravity/include/cannon_numeric.h" +##include "httpMethods/handle_HTTP_GET_hello.h" +##include "httpMethods/TimeSession.hh" + +class CannonSimObject : public Trick::SimObject { + + public: + CANNON cannon ; + int foo; + CannonSimObject() { + ("default_data") cannon_default_data( &cannon ) ; + ("initialization") cannon_init( &cannon ) ; + ("derivative") cannon_deriv( &cannon ) ; + ("integration") trick_ret = cannon_integ( &cannon ) ; + ("dynamic_event") cannon_impact( &cannon) ; + } +} ; +CannonSimObject dyn ; + +IntegLoop dyn_integloop (0.01) dyn; + +void create_connections() { + dyn_integloop.getIntegrator(Runge_Kutta_4, 5); + web.server.installHTTPGEThandler( "hello", &handle_HTTP_GET_hello ); + web.server.installWebSocketSessionMaker( "Time", &makeTimeSession ); +} \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/S_overrides.mk b/trick_sims/Cannon/SIM_cannon_webserver/S_overrides.mk new file mode 100644 index 00000000..26b2544b --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/S_overrides.mk @@ -0,0 +1,3 @@ +TRICK_CFLAGS += -I../models +TRICK_CXXFLAGS += -I../models + diff --git a/trick_sims/Cannon/SIM_cannon_webserver/images/CannonInit.png b/trick_sims/Cannon/SIM_cannon_webserver/images/CannonInit.png new file mode 100644 index 00000000..8801beed Binary files /dev/null and b/trick_sims/Cannon/SIM_cannon_webserver/images/CannonInit.png differ diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/alloc_info.html b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/alloc_info.html new file mode 100644 index 00000000..056cff03 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/alloc_info.html @@ -0,0 +1,156 @@ + + + + +Trick Memory Allocations + + + + + +
+
+
+
+ + + + + diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/TrickLogoSmall.png b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/TrickLogoSmall.png new file mode 100644 index 00000000..d6712b42 Binary files /dev/null and b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/TrickLogoSmall.png differ diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/asset-manifest.json b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/asset-manifest.json new file mode 100644 index 00000000..33338ad4 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/asset-manifest.json @@ -0,0 +1,18 @@ +{ + "files": { + "main.css": "./static/css/main.a82b4685.chunk.css", + "main.js": "./static/js/main.593ee672.chunk.js", + "main.js.map": "./static/js/main.593ee672.chunk.js.map", + "runtime~main.js": "./static/js/runtime~main.d653cc00.js", + "runtime~main.js.map": "./static/js/runtime~main.d653cc00.js.map", + "static/css/2.335e4b44.chunk.css": "./static/css/2.335e4b44.chunk.css", + "static/js/2.fd0d41cf.chunk.js": "./static/js/2.fd0d41cf.chunk.js", + "static/js/2.fd0d41cf.chunk.js.map": "./static/js/2.fd0d41cf.chunk.js.map", + "index.html": "./index.html", + "precache-manifest.9d8f7ddf3680a6a6d643dad7fa7c8492.js": "./precache-manifest.9d8f7ddf3680a6a6d643dad7fa7c8492.js", + "service-worker.js": "./service-worker.js", + "static/css/2.335e4b44.chunk.css.map": "./static/css/2.335e4b44.chunk.css.map", + "static/css/main.a82b4685.chunk.css.map": "./static/css/main.a82b4685.chunk.css.map", + "static/media/index.css": "./static/media/roboto-latin-900italic.ebf6d164.woff2" + } +} \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/index.html b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/index.html new file mode 100644 index 00000000..d4711813 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/index.html @@ -0,0 +1 @@ +React App
\ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/manifest.json b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/manifest.json new file mode 100644 index 00000000..1f2f141f --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/manifest.json @@ -0,0 +1,15 @@ +{ + "short_name": "React App", + "name": "Create React App Sample", + "icons": [ + { + "src": "favicon.ico", + "sizes": "64x64 32x32 24x24 16x16", + "type": "image/x-icon" + } + ], + "start_url": ".", + "display": "standalone", + "theme_color": "#000000", + "background_color": "#ffffff" +} diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/precache-manifest.9d8f7ddf3680a6a6d643dad7fa7c8492.js b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/precache-manifest.9d8f7ddf3680a6a6d643dad7fa7c8492.js new file mode 100644 index 00000000..b4b8c165 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/precache-manifest.9d8f7ddf3680a6a6d643dad7fa7c8492.js @@ -0,0 +1,122 @@ +self.__precacheManifest = (self.__precacheManifest || []).concat([ + { + "revision": "577c8ea84fd85c60dc5f5c579560e83d", + "url": "./index.html" + }, + { + "revision": "6c79f9b2aedb002d302c", + "url": "./static/css/2.335e4b44.chunk.css" + }, + { + "revision": "9b46f0f8321ce396aba6", + "url": "./static/css/main.a82b4685.chunk.css" + }, + { + "revision": "6c79f9b2aedb002d302c", + "url": "./static/js/2.fd0d41cf.chunk.js" + }, + { + "revision": "9b46f0f8321ce396aba6", + "url": "./static/js/main.593ee672.chunk.js" + }, + { + "revision": "8c97409f0ee389fe75da", + "url": "./static/js/runtime~main.d653cc00.js" + }, + { + "revision": "5cb7edfceb233100075dc9a1e12e8da3", + "url": "./static/media/roboto-latin-100.5cb7edfc.woff" + }, + { + "revision": "7370c3679472e9560965ff48a4399d0b", + "url": "./static/media/roboto-latin-100.7370c367.woff2" + }, + { + "revision": "f8b1df51ba843179fa1cc9b53d58127a", + "url": "./static/media/roboto-latin-100italic.f8b1df51.woff2" + }, + { + "revision": "f9e8e590b4e0f1ff83469bb2a55b8488", + "url": "./static/media/roboto-latin-100italic.f9e8e590.woff" + }, + { + "revision": "b00849e00f4c2331cddd8ffb44a6720b", + "url": "./static/media/roboto-latin-300.b00849e0.woff" + }, + { + "revision": "ef7c6637c68f269a882e73bcb57a7f6a", + "url": "./static/media/roboto-latin-300.ef7c6637.woff2" + }, + { + "revision": "14286f3ba79c6627433572dfa925202e", + "url": "./static/media/roboto-latin-300italic.14286f3b.woff2" + }, + { + "revision": "4df32891a5f2f98a363314f595482e08", + "url": "./static/media/roboto-latin-300italic.4df32891.woff" + }, + { + "revision": "479970ffb74f2117317f9d24d9e317fe", + "url": "./static/media/roboto-latin-400.479970ff.woff2" + }, + { + "revision": "60fa3c0614b8fb2f394fa29944c21540", + "url": "./static/media/roboto-latin-400.60fa3c06.woff" + }, + { + "revision": "51521a2a8da71e50d871ac6fd2187e87", + "url": "./static/media/roboto-latin-400italic.51521a2a.woff2" + }, + { + "revision": "fe65b8335ee19dd944289f9ed3178c78", + "url": "./static/media/roboto-latin-400italic.fe65b833.woff" + }, + { + "revision": "020c97dc8e0463259c2f9df929bb0c69", + "url": "./static/media/roboto-latin-500.020c97dc.woff2" + }, + { + "revision": "87284894879f5b1c229cb49c8ff6decc", + "url": "./static/media/roboto-latin-500.87284894.woff" + }, + { + "revision": "288ad9c6e8b43cf02443a1f499bdf67e", + "url": "./static/media/roboto-latin-500italic.288ad9c6.woff" + }, + { + "revision": "db4a2a231f52e497c0191e8966b0ee58", + "url": "./static/media/roboto-latin-500italic.db4a2a23.woff2" + }, + { + "revision": "2735a3a69b509faf3577afd25bdf552e", + "url": "./static/media/roboto-latin-700.2735a3a6.woff2" + }, + { + "revision": "adcde98f1d584de52060ad7b16373da3", + "url": "./static/media/roboto-latin-700.adcde98f.woff" + }, + { + "revision": "81f57861ed4ac74741f5671e1dff2fd9", + "url": "./static/media/roboto-latin-700italic.81f57861.woff" + }, + { + "revision": "da0e717829e033a69dec97f1e155ae42", + "url": "./static/media/roboto-latin-700italic.da0e7178.woff2" + }, + { + "revision": "9b3766ef4a402ad3fdeef7501a456512", + "url": "./static/media/roboto-latin-900.9b3766ef.woff2" + }, + { + "revision": "bb1e4dc6333675d11ada2e857e7f95d7", + "url": "./static/media/roboto-latin-900.bb1e4dc6.woff" + }, + { + "revision": "28f9151055c950874d2c6803a39b425b", + "url": "./static/media/roboto-latin-900italic.28f91510.woff" + }, + { + "revision": "ebf6d1640ccddb99fb49f73c052c55a8", + "url": "./static/media/roboto-latin-900italic.ebf6d164.woff2" + } +]); \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/service-worker.js b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/service-worker.js new file mode 100644 index 00000000..b5113d07 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/service-worker.js @@ -0,0 +1,39 @@ +/** + * Welcome to your Workbox-powered service worker! + * + * You'll need to register this file in your web app and you should + * disable HTTP caching for this file too. + * See https://goo.gl/nhQhGp + * + * The rest of the code is auto-generated. Please don't update this file + * directly; instead, make changes to your Workbox build configuration + * and re-run your build process. + * See https://goo.gl/2aRDsh + */ + +importScripts("https://storage.googleapis.com/workbox-cdn/releases/4.3.1/workbox-sw.js"); + +importScripts( + "./precache-manifest.9d8f7ddf3680a6a6d643dad7fa7c8492.js" +); + +self.addEventListener('message', (event) => { + if (event.data && event.data.type === 'SKIP_WAITING') { + self.skipWaiting(); + } +}); + +workbox.core.clientsClaim(); + +/** + * The workboxSW.precacheAndRoute() method efficiently caches and responds to + * requests for URLs in the manifest. + * See https://goo.gl/S9QRab + */ +self.__precacheManifest = [].concat(self.__precacheManifest || []); +workbox.precaching.precacheAndRoute(self.__precacheManifest, {}); + +workbox.routing.registerNavigationRoute(workbox.precaching.getCacheKeyForURL("./index.html"), { + + blacklist: [/^\/_/,/\/[^\/]+\.[^\/]+$/], +}); diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/2.335e4b44.chunk.css b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/2.335e4b44.chunk.css new file mode 100644 index 00000000..822243b2 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/2.335e4b44.chunk.css @@ -0,0 +1,2 @@ +@font-face{font-family:Roboto;font-style:normal;font-display:swap;font-weight:100;src:local("Roboto Thin "),local("Roboto-Thin"),url(../../static/media/roboto-latin-100.7370c367.woff2) format("woff2"),url(../../static/media/roboto-latin-100.5cb7edfc.woff) format("woff")}@font-face{font-family:Roboto;font-style:italic;font-display:swap;font-weight:100;src:local("Roboto Thin italic"),local("Roboto-Thinitalic"),url(../../static/media/roboto-latin-100italic.f8b1df51.woff2) format("woff2"),url(../../static/media/roboto-latin-100italic.f9e8e590.woff) format("woff")}@font-face{font-family:Roboto;font-style:normal;font-display:swap;font-weight:300;src:local("Roboto Light "),local("Roboto-Light"),url(../../static/media/roboto-latin-300.ef7c6637.woff2) format("woff2"),url(../../static/media/roboto-latin-300.b00849e0.woff) format("woff")}@font-face{font-family:Roboto;font-style:italic;font-display:swap;font-weight:300;src:local("Roboto Light italic"),local("Roboto-Lightitalic"),url(../../static/media/roboto-latin-300italic.14286f3b.woff2) format("woff2"),url(../../static/media/roboto-latin-300italic.4df32891.woff) format("woff")}@font-face{font-family:Roboto;font-style:normal;font-display:swap;font-weight:400;src:local("Roboto Regular "),local("Roboto-Regular"),url(../../static/media/roboto-latin-400.479970ff.woff2) format("woff2"),url(../../static/media/roboto-latin-400.60fa3c06.woff) format("woff")}@font-face{font-family:Roboto;font-style:italic;font-display:swap;font-weight:400;src:local("Roboto Regular italic"),local("Roboto-Regularitalic"),url(../../static/media/roboto-latin-400italic.51521a2a.woff2) format("woff2"),url(../../static/media/roboto-latin-400italic.fe65b833.woff) format("woff")}@font-face{font-family:Roboto;font-style:normal;font-display:swap;font-weight:500;src:local("Roboto Medium "),local("Roboto-Medium"),url(../../static/media/roboto-latin-500.020c97dc.woff2) format("woff2"),url(../../static/media/roboto-latin-500.87284894.woff) format("woff")}@font-face{font-family:Roboto;font-style:italic;font-display:swap;font-weight:500;src:local("Roboto Medium italic"),local("Roboto-Mediumitalic"),url(../../static/media/roboto-latin-500italic.db4a2a23.woff2) format("woff2"),url(../../static/media/roboto-latin-500italic.288ad9c6.woff) format("woff")}@font-face{font-family:Roboto;font-style:normal;font-display:swap;font-weight:700;src:local("Roboto Bold "),local("Roboto-Bold"),url(../../static/media/roboto-latin-700.2735a3a6.woff2) format("woff2"),url(../../static/media/roboto-latin-700.adcde98f.woff) format("woff")}@font-face{font-family:Roboto;font-style:italic;font-display:swap;font-weight:700;src:local("Roboto Bold italic"),local("Roboto-Bolditalic"),url(../../static/media/roboto-latin-700italic.da0e7178.woff2) format("woff2"),url(../../static/media/roboto-latin-700italic.81f57861.woff) format("woff")}@font-face{font-family:Roboto;font-style:normal;font-display:swap;font-weight:900;src:local("Roboto Black "),local("Roboto-Black"),url(../../static/media/roboto-latin-900.9b3766ef.woff2) format("woff2"),url(../../static/media/roboto-latin-900.bb1e4dc6.woff) format("woff")}@font-face{font-family:Roboto;font-style:italic;font-display:swap;font-weight:900;src:local("Roboto Black italic"),local("Roboto-Blackitalic"),url(../../static/media/roboto-latin-900italic.ebf6d164.woff2) format("woff2"),url(../../static/media/roboto-latin-900italic.28f91510.woff) format("woff")} +/*# sourceMappingURL=2.335e4b44.chunk.css.map */ \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/2.335e4b44.chunk.css.map b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/2.335e4b44.chunk.css.map new file mode 100644 index 00000000..30d6dadf --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/2.335e4b44.chunk.css.map @@ -0,0 +1 @@ +{"version":3,"sources":["index.css"],"names":[],"mappings":"AACA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,4LAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,oNAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,8LAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,sNAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,kMAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,0NAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,gMAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,wNAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,4LAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,oNAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,8LAKF,CAGA,WACE,kBAAqB,CACrB,iBAAkB,CAClB,iBAAkB,CAClB,eAAgB,CAChB,sNAKF","file":"2.335e4b44.chunk.css","sourcesContent":["/* roboto-100normal - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: normal;\n font-display: swap;\n font-weight: 100;\n src:\n local('Roboto Thin '),\n local('Roboto-Thin'),\n url('./files/roboto-latin-100.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-100.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-100italic - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: italic;\n font-display: swap;\n font-weight: 100;\n src:\n local('Roboto Thin italic'),\n local('Roboto-Thinitalic'),\n url('./files/roboto-latin-100italic.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-100italic.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-300normal - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: normal;\n font-display: swap;\n font-weight: 300;\n src:\n local('Roboto Light '),\n local('Roboto-Light'),\n url('./files/roboto-latin-300.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-300.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-300italic - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: italic;\n font-display: swap;\n font-weight: 300;\n src:\n local('Roboto Light italic'),\n local('Roboto-Lightitalic'),\n url('./files/roboto-latin-300italic.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-300italic.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-400normal - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: normal;\n font-display: swap;\n font-weight: 400;\n src:\n local('Roboto Regular '),\n local('Roboto-Regular'),\n url('./files/roboto-latin-400.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-400.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-400italic - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: italic;\n font-display: swap;\n font-weight: 400;\n src:\n local('Roboto Regular italic'),\n local('Roboto-Regularitalic'),\n url('./files/roboto-latin-400italic.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-400italic.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-500normal - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: normal;\n font-display: swap;\n font-weight: 500;\n src:\n local('Roboto Medium '),\n local('Roboto-Medium'),\n url('./files/roboto-latin-500.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-500.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-500italic - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: italic;\n font-display: swap;\n font-weight: 500;\n src:\n local('Roboto Medium italic'),\n local('Roboto-Mediumitalic'),\n url('./files/roboto-latin-500italic.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-500italic.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-700normal - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: normal;\n font-display: swap;\n font-weight: 700;\n src:\n local('Roboto Bold '),\n local('Roboto-Bold'),\n url('./files/roboto-latin-700.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-700.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-700italic - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: italic;\n font-display: swap;\n font-weight: 700;\n src:\n local('Roboto Bold italic'),\n local('Roboto-Bolditalic'),\n url('./files/roboto-latin-700italic.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-700italic.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-900normal - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: normal;\n font-display: swap;\n font-weight: 900;\n src:\n local('Roboto Black '),\n local('Roboto-Black'),\n url('./files/roboto-latin-900.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-900.woff') format('woff'); /* Modern Browsers */\n}\n\n/* roboto-900italic - latin */\n@font-face {\n font-family: 'Roboto';\n font-style: italic;\n font-display: swap;\n font-weight: 900;\n src:\n local('Roboto Black italic'),\n local('Roboto-Blackitalic'),\n url('./files/roboto-latin-900italic.woff2') format('woff2'), /* Super Modern Browsers */\n url('./files/roboto-latin-900italic.woff') format('woff'); /* Modern Browsers */\n}\n\n"]} \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/main.a82b4685.chunk.css b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/main.a82b4685.chunk.css new file mode 100644 index 00000000..93b61525 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/main.a82b4685.chunk.css @@ -0,0 +1,2 @@ +body{margin:0;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,Fira Sans,Droid Sans,Helvetica Neue,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}code{font-family:source-code-pro,Menlo,Monaco,Consolas,Courier New,monospace} +/*# sourceMappingURL=main.a82b4685.chunk.css.map */ \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/main.a82b4685.chunk.css.map b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/main.a82b4685.chunk.css.map new file mode 100644 index 00000000..dea0a0c4 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/css/main.a82b4685.chunk.css.map @@ -0,0 +1 @@ +{"version":3,"sources":["index.css"],"names":[],"mappings":"AACA,KACE,QAAS,CACT,mIAEY,CACZ,kCAAmC,CACnC,iCACF,CAEA,KACE,uEAEF","file":"main.a82b4685.chunk.css","sourcesContent":["\nbody {\n margin: 0;\n font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", \"Roboto\", \"Oxygen\",\n \"Ubuntu\", \"Cantarell\", \"Fira Sans\", \"Droid Sans\", \"Helvetica Neue\",\n sans-serif;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n}\n\ncode {\n font-family: source-code-pro, Menlo, Monaco, Consolas, \"Courier New\",\n monospace;\n}\n"]} \ No newline at end of file diff --git a/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/js/2.fd0d41cf.chunk.js b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/js/2.fd0d41cf.chunk.js new file mode 100644 index 00000000..b7bf1f10 --- /dev/null +++ b/trick_sims/Cannon/SIM_cannon_webserver/www/apps/react/static/js/2.fd0d41cf.chunk.js @@ -0,0 +1,2 @@ +(window.webpackJsonp=window.webpackJsonp||[]).push([[2],[function(e,t,n){"use strict";e.exports=n(87)},function(e,t,n){"use strict";function r(){return(r=Object.assign||function(e){for(var t=1;t=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(a[n]=e[n])}return a}},function(e,t,n){"use strict";function r(e){var t,n,o="";if(e)if("object"===typeof e)if(e.push)for(t=0;t1&&void 0!==arguments[1]?arguments[1]:{};return function(n){var a=t.defaultTheme,l=t.withTheme,f=void 0!==l&&l,p=t.name,h=Object(o.a)(t,["defaultTheme","withTheme","name"]),m=p,v=Object(s.a)(e,Object(r.a)({defaultTheme:a,Component:n,name:p||n.displayName,classNamePrefix:m},h)),b=i.a.forwardRef(function(e,t){e.classes;var l,u=e.innerRef,s=Object(o.a)(e,["classes","innerRef"]),h=v(e),m=s;return("string"===typeof p||f)&&(l=Object(d.a)()||a,p&&(m=c({theme:l,name:p,props:s})),f&&!m.theme&&(m.theme=l)),i.a.createElement(n,Object(r.a)({ref:u||t,classes:h},m))});return u()(b,n),b}},p=n(25);t.a=function(e,t){return f(e,Object(r.a)({defaultTheme:p.a},t))}},function(e,t,n){e.exports=n(96)()},function(e,t,n){"use strict";n.d(t,"a",function(){return a}),n.d(t,"b",function(){return i}),n.d(t,"c",function(){return l});var r=n(0),o=n.n(r);function a(e,t){return o.a.isValidElement(e)&&-1!==t.indexOf(e.type.muiName)}function i(e,t){"function"===typeof e?e(t):e&&(e.current=t)}function l(e,t){return o.a.useMemo(function(){return null==e&&null==t?null:function(n){i(e,n),i(t,n)}},[e,t])}},function(e,t,n){"use strict";n.d(t,"a",function(){return r}),n.d(t,"b",function(){return o});n(10);function r(e){return e.charAt(0).toUpperCase()+e.slice(1)}function o(){for(var e=arguments.length,t=new Array(e),n=0;n1&&void 0!==arguments[1]?arguments[1]:0,n=arguments.length>2&&void 0!==arguments[2]?arguments[2]:1;return en?n:e}function o(e){if(e.type)return e;if("#"===e.charAt(0))return o(function(e){e=e.substr(1);var t=new RegExp(".{1,".concat(e.length/3,"}"),"g"),n=e.match(t);return n&&1===n[0].length&&(n=n.map(function(e){return e+e})),n?"rgb(".concat(n.map(function(e){return parseInt(e,16)}).join(", "),")"):""}(e));var t=e.indexOf("("),n=e.substring(0,t);if(-1===["rgb","rgba","hsl","hsla"].indexOf(n))throw new Error(["Material-UI: unsupported `".concat(e,"` color."),"We support the following formats: #nnn, #nnnnnn, rgb(), rgba(), hsl(), hsla()."].join("\n"));var r=e.substring(t+1,e.length-1).split(",");return{type:n,values:r=r.map(function(e){return parseFloat(e)})}}function a(e){var t=e.type,n=e.values;return-1!==t.indexOf("rgb")?n=n.map(function(e,t){return t<3?parseInt(e,10):e}):-1!==t.indexOf("hsl")&&(n[1]="".concat(n[1],"%"),n[2]="".concat(n[2],"%")),"".concat(t,"(").concat(n.join(", "),")")}function i(e,t){var n=l(e),r=l(t);return(Math.max(n,r)+.05)/(Math.min(n,r)+.05)}function l(e){var t="hsl"===(e=o(e)).type?o(function(e){var t=(e=o(e)).values,n=t[0],r=t[1]/100,i=t[2]/100,l=r*Math.min(i,1-i),u=function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:(e+n/30)%12;return i-l*Math.max(Math.min(t-3,9-t,1),-1)},s="rgb",c=[Math.round(255*u(0)),Math.round(255*u(8)),Math.round(255*u(4))];return"hsla"===e.type&&(s+="a",c.push(t[3])),a({type:s,values:c})}(e)).values:e.values;return t=t.map(function(e){return(e/=255)<=.03928?e/12.92:Math.pow((e+.055)/1.055,2.4)}),Number((.2126*t[0]+.7152*t[1]+.0722*t[2]).toFixed(3))}function u(e,t){return e=o(e),t=r(t),"rgb"!==e.type&&"hsl"!==e.type||(e.type+="a"),e.values[3]=t,a(e)}function s(e,t){if(e=o(e),t=r(t),-1!==e.type.indexOf("hsl"))e.values[2]*=1-t;else if(-1!==e.type.indexOf("rgb"))for(var n=0;n<3;n+=1)e.values[n]*=1-t;return a(e)}function c(e,t){if(e=o(e),t=r(t),-1!==e.type.indexOf("hsl"))e.values[2]+=(100-e.values[2])*t;else if(-1!==e.type.indexOf("rgb"))for(var n=0;n<3;n+=1)e.values[n]+=(255-e.values[n])*t;return a(e)}},function(e,t,n){"use strict";n.r(t);var r=n(17),o=n.n(r),a=function(e,t){var n=o()({},e,t);for(var r in e){var a;e[r]&&"object"===typeof t[r]&&o()(n,((a={})[r]=o()(e[r],t[r]),a))}return n},i={breakpoints:[40,52,64].map(function(e){return e+"em"})},l=function(e){return"@media screen and (min-width: "+e+")"},u=function(e,t){return s(t,e,e)},s=function(e,t,n,r,o){for(t=t&&t.split?t.split("."):[t],r=0;r1&&u.forEach(function(n){var o;r[n]=e(((o={})[n]=t[n],o))}),r},d=function(e,t,n,r){var a={};return r.slice(0,e.length).forEach(function(r,i){var l,u=e[i],s=t(r,n);u?o()(a,((l={})[u]=o()({},a[u],s),l)):o()(a,s)}),a},f=function(e,t,n,r){var a={};for(var i in r){var u=e[i],s=t(r[i],n);if(u){var c,d=l(u);o()(a,((c={})[d]=o()({},a[d],s),c))}else o()(a,s)}return a},p=function(e){var t=e.properties,n=e.property,r=e.scale,o=e.transform,a=void 0===o?u:o,i=e.defaultScale;t=t||[n];var l=function(e,n){var r={},o=a(e,n);if(null!==o)return t.forEach(function(e){r[e]=o}),r};return l.scale=r,l.defaults=i,l},h=function(e){void 0===e&&(e={});var t={};return Object.keys(e).forEach(function(n){var r=e[n];t[n]=!0!==r?"function"!==typeof r?p(r):r:p({property:n,scale:n})}),c(t)},m=function(){for(var e={},t=arguments.length,n=new Array(t),r=0;r1?e:100*e+"%")}},height:{property:"height",scale:"sizes"},minWidth:{property:"minWidth",scale:"sizes"},minHeight:{property:"minHeight",scale:"sizes"},maxWidth:{property:"maxWidth",scale:"sizes"},maxHeight:{property:"maxHeight",scale:"sizes"},size:{properties:["width","height"],scale:"sizes"},overflow:!0,display:!0,verticalAlign:!0}),b=v,g={color:{property:"color",scale:"colors"},backgroundColor:{property:"backgroundColor",scale:"colors"},opacity:!0};g.bg=g.backgroundColor;var y=h(g),x=y,k=h({fontFamily:{property:"fontFamily",scale:"fonts"},fontSize:{property:"fontSize",scale:"fontSizes",defaultScale:[12,14,16,20,24,32,48,64,72]},fontWeight:{property:"fontWeight",scale:"fontWeights"},lineHeight:{property:"lineHeight",scale:"lineHeights"},letterSpacing:{property:"letterSpacing",scale:"letterSpacings"},textAlign:!0,fontStyle:!0}),w=k,O=h({alignItems:!0,alignContent:!0,justifyItems:!0,justifyContent:!0,flexWrap:!0,flexDirection:!0,flex:!0,flexGrow:!0,flexShrink:!0,flexBasis:!0,justifySelf:!0,alignSelf:!0,order:!0}),S=O,C={space:[0,4,8,16,32,64,128,256,512]},E=h({gridGap:{property:"gridGap",scale:"space",defaultScale:C.space},gridColumnGap:{property:"gridColumnGap",scale:"space",defaultScale:C.space},gridRowGap:{property:"gridRowGap",scale:"space",defaultScale:C.space},gridColumn:!0,gridRow:!0,gridAutoFlow:!0,gridAutoColumns:!0,gridAutoRows:!0,gridTemplateColumns:!0,gridTemplateRows:!0,gridTemplateAreas:!0,gridArea:!0}),j=E,T=h({border:{property:"border",scale:"borders"},borderWidth:{property:"borderWidth",scale:"borderWidths"},borderStyle:{property:"borderStyle",scale:"borderStyles"},borderColor:{property:"borderColor",scale:"colors"},borderRadius:{property:"borderRadius",scale:"radii"},borderTop:{property:"borderTop",scale:"borders"},borderRight:{property:"borderRight",scale:"borders"},borderBottom:{property:"borderBottom",scale:"borders"},borderLeft:{property:"borderLeft",scale:"borders"},borderX:{properties:["borderLeft","borderRight"],scale:"borders"},borderY:{properties:["borderTop","borderBottom"],scale:"borders"}}),P=T,R={background:!0,backgroundImage:!0,backgroundSize:!0,backgroundPosition:!0,backgroundRepeat:!0};R.bgImage=R.backgroundImage,R.bgSize=R.backgroundSize,R.bgPosition=R.backgroundPosition,R.bgRepeat=R.backgroundRepeat;var N=h(R),_=N,A={space:[0,4,8,16,32,64,128,256,512]},M=h({position:!0,zIndex:{property:"zIndex",scale:"zIndices"},top:{property:"top",scale:"space",defaultScale:A.space},right:{property:"right",scale:"space",defaultScale:A.space},bottom:{property:"bottom",scale:"space",defaultScale:A.space},left:{property:"left",scale:"space",defaultScale:A.space}}),I=M,z={space:[0,4,8,16,32,64,128,256,512]},D=function(e){return"number"===typeof e&&!isNaN(e)},L=function(e,t){if(!D(e))return s(t,e,e);var n=e<0,r=Math.abs(e),o=s(t,r,r);return D(o)?o*(n?-1:1):n?"-"+o:o},F={};F.margin={margin:{property:"margin",scale:"space",transform:L,defaultScale:z.space},marginTop:{property:"marginTop",scale:"space",transform:L,defaultScale:z.space},marginRight:{property:"marginRight",scale:"space",transform:L,defaultScale:z.space},marginBottom:{property:"marginBottom",scale:"space",transform:L,defaultScale:z.space},marginLeft:{property:"marginLeft",scale:"space",transform:L,defaultScale:z.space},marginX:{properties:["marginLeft","marginRight"],scale:"space",transform:L,defaultScale:z.space},marginY:{properties:["marginTop","marginBottom"],scale:"space",transform:L,defaultScale:z.space}},F.margin.m=F.margin.margin,F.margin.mt=F.margin.marginTop,F.margin.mr=F.margin.marginRight,F.margin.mb=F.margin.marginBottom,F.margin.ml=F.margin.marginLeft,F.margin.mx=F.margin.marginX,F.margin.my=F.margin.marginY,F.padding={padding:{property:"padding",scale:"space",defaultScale:z.space},paddingTop:{property:"paddingTop",scale:"space",defaultScale:z.space},paddingRight:{property:"paddingRight",scale:"space",defaultScale:z.space},paddingBottom:{property:"paddingBottom",scale:"space",defaultScale:z.space},paddingLeft:{property:"paddingLeft",scale:"space",defaultScale:z.space},paddingX:{properties:["paddingLeft","paddingRight"],scale:"space",defaultScale:z.space},paddingY:{properties:["paddingTop","paddingBottom"],scale:"space",defaultScale:z.space}},F.padding.p=F.padding.padding,F.padding.pt=F.padding.paddingTop,F.padding.pr=F.padding.paddingRight,F.padding.pb=F.padding.paddingBottom,F.padding.pl=F.padding.paddingLeft,F.padding.px=F.padding.paddingX,F.padding.py=F.padding.paddingY;var W=h(F.margin),B=h(F.padding),U=m(W,B),H=h({boxShadow:{property:"boxShadow",scale:"shadows"},textShadow:{property:"textShadow",scale:"shadows"}}),$=H,V=function(e){var t,n=e.scale,r=e.prop,o=void 0===r?"variant":r,a=e.key,i=function(e,t){return s(t,e,null)};i.scale=n||a;var l=((t={})[o]=i,t);return c(l)},q=V({key:"buttons"}),K=V({key:"textStyles",prop:"textStyle"}),X=V({key:"colorStyles",prop:"colors"});n.d(t,"width",function(){return G}),n.d(t,"height",function(){return Y}),n.d(t,"minWidth",function(){return Q}),n.d(t,"minHeight",function(){return J}),n.d(t,"maxWidth",function(){return Z}),n.d(t,"maxHeight",function(){return ee}),n.d(t,"size",function(){return te}),n.d(t,"verticalAlign",function(){return ne}),n.d(t,"display",function(){return re}),n.d(t,"overflow",function(){return oe}),n.d(t,"opacity",function(){return ae}),n.d(t,"fontSize",function(){return ie}),n.d(t,"fontFamily",function(){return le}),n.d(t,"fontWeight",function(){return ue}),n.d(t,"lineHeight",function(){return se}),n.d(t,"textAlign",function(){return ce}),n.d(t,"fontStyle",function(){return de}),n.d(t,"letterSpacing",function(){return fe}),n.d(t,"alignItems",function(){return pe}),n.d(t,"alignContent",function(){return he}),n.d(t,"justifyItems",function(){return me}),n.d(t,"justifyContent",function(){return ve}),n.d(t,"flexWrap",function(){return be}),n.d(t,"flexDirection",function(){return ge}),n.d(t,"flex",function(){return ye}),n.d(t,"flexGrow",function(){return xe}),n.d(t,"flexShrink",function(){return ke}),n.d(t,"flexBasis",function(){return we}),n.d(t,"justifySelf",function(){return Oe}),n.d(t,"alignSelf",function(){return Se}),n.d(t,"order",function(){return Ce}),n.d(t,"gridGap",function(){return Ee}),n.d(t,"gridColumnGap",function(){return je}),n.d(t,"gridRowGap",function(){return Te}),n.d(t,"gridColumn",function(){return Pe}),n.d(t,"gridRow",function(){return Re}),n.d(t,"gridAutoFlow",function(){return Ne}),n.d(t,"gridAutoColumns",function(){return _e}),n.d(t,"gridAutoRows",function(){return Ae}),n.d(t,"gridTemplateColumns",function(){return Me}),n.d(t,"gridTemplateRows",function(){return Ie}),n.d(t,"gridTemplateAreas",function(){return ze}),n.d(t,"gridArea",function(){return De}),n.d(t,"borderWidth",function(){return Le}),n.d(t,"borderStyle",function(){return Fe}),n.d(t,"borderColor",function(){return We}),n.d(t,"borderTop",function(){return Be}),n.d(t,"borderRight",function(){return Ue}),n.d(t,"borderBottom",function(){return He}),n.d(t,"borderLeft",function(){return $e}),n.d(t,"borderRadius",function(){return Ve}),n.d(t,"backgroundImage",function(){return qe}),n.d(t,"backgroundSize",function(){return Ke}),n.d(t,"backgroundPosition",function(){return Xe}),n.d(t,"backgroundRepeat",function(){return Ge}),n.d(t,"zIndex",function(){return Ye}),n.d(t,"top",function(){return Qe}),n.d(t,"right",function(){return Je}),n.d(t,"bottom",function(){return Ze}),n.d(t,"left",function(){return et}),n.d(t,"style",function(){return tt}),n.d(t,"get",function(){return s}),n.d(t,"createParser",function(){return c}),n.d(t,"createStyleFunction",function(){return p}),n.d(t,"compose",function(){return m}),n.d(t,"system",function(){return h}),n.d(t,"margin",function(){return W}),n.d(t,"padding",function(){return B}),n.d(t,"space",function(){return U}),n.d(t,"color",function(){return y}),n.d(t,"layout",function(){return v}),n.d(t,"typography",function(){return k}),n.d(t,"flexbox",function(){return O}),n.d(t,"border",function(){return T}),n.d(t,"background",function(){return N}),n.d(t,"position",function(){return M}),n.d(t,"grid",function(){return E}),n.d(t,"shadow",function(){return H}),n.d(t,"boxShadow",function(){return $}),n.d(t,"textShadow",function(){return $}),n.d(t,"variant",function(){return V}),n.d(t,"buttonStyle",function(){return q}),n.d(t,"textStyle",function(){return K}),n.d(t,"colorStyle",function(){return X}),n.d(t,"borders",function(){return P});var G=b.width,Y=b.height,Q=b.minWidth,J=b.minHeight,Z=b.maxWidth,ee=b.maxHeight,te=b.size,ne=b.verticalAlign,re=b.display,oe=b.overflow,ae=x.opacity,ie=w.fontSize,le=w.fontFamily,ue=w.fontWeight,se=w.lineHeight,ce=w.textAlign,de=w.fontStyle,fe=w.letterSpacing,pe=S.alignItems,he=S.alignContent,me=S.justifyItems,ve=S.justifyContent,be=S.flexWrap,ge=S.flexDirection,ye=S.flex,xe=S.flexGrow,ke=S.flexShrink,we=S.flexBasis,Oe=S.justifySelf,Se=S.alignSelf,Ce=S.order,Ee=j.gridGap,je=j.gridColumnGap,Te=j.gridRowGap,Pe=j.gridColumn,Re=j.gridRow,Ne=j.gridAutoFlow,_e=j.gridAutoColumns,Ae=j.gridAutoRows,Me=j.gridTemplateColumns,Ie=j.gridTemplateRows,ze=j.gridTemplateAreas,De=j.gridArea,Le=P.borderWidth,Fe=P.borderStyle,We=P.borderColor,Be=P.borderTop,Ue=P.borderRight,He=P.borderBottom,$e=P.borderLeft,Ve=P.borderRadius,qe=_.backgroundImage,Ke=_.backgroundSize,Xe=_.backgroundPosition,Ge=_.backgroundRepeat,Ye=I.zIndex,Qe=I.top,Je=I.right,Ze=I.bottom,et=I.left,tt=function(e){var t=e.prop,n=e.cssProperty,r=e.alias,o=e.key,a=e.transformValue,i=e.scale,l=e.properties,u={};return u[t]=p({properties:l,property:n||t,scale:o,defaultScale:i,transform:a}),r&&(u[r]=u[t]),c(u)}},function(e,t,n){"use strict";n.d(t,"b",function(){return r}),n.d(t,"a",function(){return o});var r=function(e){return e.scrollTop};function o(e,t){var n=e.timeout,r=e.style,o=void 0===r?{}:r;return{duration:o.transitionDuration||"number"===typeof n?n:n[t.mode]||0,delay:o.transitionDelay}}},,function(e,t,n){"use strict";n.d(t,"a",function(){return a});var r=n(104),o=n(25);function a(){return Object(r.a)()||o.a}},function(e,t,n){"use strict";t.a=function(e){return e&&e.ownerDocument||document}},function(e,t,n){"use strict";var r=Object.getOwnPropertySymbols,o=Object.prototype.hasOwnProperty,a=Object.prototype.propertyIsEnumerable;e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var t={},n=0;n<10;n++)t["_"+String.fromCharCode(n)]=n;if("0123456789"!==Object.getOwnPropertyNames(t).map(function(e){return t[e]}).join(""))return!1;var r={};return"abcdefghijklmnopqrst".split("").forEach(function(e){r[e]=e}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},r)).join("")}catch(o){return!1}}()?Object.assign:function(e,t){for(var n,i,l=function(e){if(null===e||void 0===e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}(e),u=1;u0&&void 0!==arguments[0]?arguments[0]:["all"],t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n=t.duration,l=void 0===n?a.standard:n,u=t.easing,s=void 0===u?o.easeInOut:u,c=t.delay,d=void 0===c?0:c;Object(r.a)(t,["duration","easing","delay"]);return(Array.isArray(e)?e:[e]).map(function(e){return"".concat(e," ").concat("string"===typeof l?l:i(l)," ").concat(s," ").concat("string"===typeof d?d:i(d))}).join(",")},getAutoHeightDuration:function(e){if(!e)return 0;var t=e/36;return Math.round(10*(4+15*Math.pow(t,.25)+t/5))}}},function(e,t,n){"use strict";function r(e){var t=e.props,n=e.states,r=e.muiFormControl;return n.reduce(function(e,n){return e[n]=t[n],r&&"undefined"===typeof t[n]&&(e[n]=r[n]),e},{})}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";var r=function(e){return function(e){return!!e&&"object"===typeof e}(e)&&!function(e){var t=Object.prototype.toString.call(e);return"[object RegExp]"===t||"[object Date]"===t||function(e){return e.$$typeof===o}(e)}(e)};var o="function"===typeof Symbol&&Symbol.for?Symbol.for("react.element"):60103;function a(e,t){return!1!==t.clone&&t.isMergeableObject(e)?s((n=e,Array.isArray(n)?[]:{}),e,t):e;var n}function i(e,t,n){return e.concat(t).map(function(e){return a(e,n)})}function l(e){return Object.keys(e).concat(function(e){return Object.getOwnPropertySymbols?Object.getOwnPropertySymbols(e).filter(function(t){return e.propertyIsEnumerable(t)}):[]}(e))}function u(e,t,n){var r={};return n.isMergeableObject(e)&&l(e).forEach(function(t){r[t]=a(e[t],n)}),l(t).forEach(function(o){n.isMergeableObject(t[o])&&e[o]?r[o]=function(e,t){if(!t.customMerge)return s;var n=t.customMerge(e);return"function"===typeof n?n:s}(o,n)(e[o],t[o],n):r[o]=a(t[o],n)}),r}function s(e,t,n){(n=n||{}).arrayMerge=n.arrayMerge||i,n.isMergeableObject=n.isMergeableObject||r;var o=Array.isArray(t);return o===Array.isArray(e)?o?n.arrayMerge(e,t,n):u(e,t,n):a(t,n)}s.all=function(e,t){if(!Array.isArray(e))throw new Error("first argument should be an array");return e.reduce(function(e,n){return s(e,n,t)},{})};var c=s;e.exports=c},function(e,t,n){"use strict";var r=n(0),o=n.n(r).a.createContext({});t.a=o},function(e,t,n){"use strict";n.d(t,"a",function(){return i});var r=n(0),o=n.n(r),a="undefined"!==typeof window?o.a.useLayoutEffect:o.a.useEffect;function i(e){var t=o.a.useRef(e);return a(function(){t.current=e}),o.a.useCallback(function(e){return(0,t.current)(e)},[])}},function(e,t,n){"use strict";function r(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";var r=n(77),o=Object(r.a)();t.a=o},function(e,t,n){"use strict";var r=n(0),o=n.n(r).a.createContext();t.a=o},function(e,t,n){"use strict";n.d(t,"b",function(){return i});var r=n(0),o=n.n(r),a=o.a.createContext();function i(){return o.a.useContext(a)}t.a=a},function(e,t,n){"use strict";function r(e){return null!=e&&!(Array.isArray(e)&&0===e.length)}function o(e){var t=arguments.length>1&&void 0!==arguments[1]&&arguments[1];return e&&(r(e.value)&&""!==e.value||t&&r(e.defaultValue)&&""!==e.defaultValue)}function a(e){return e.startAdornment}n.d(t,"b",function(){return o}),n.d(t,"a",function(){return a})},function(e,t,n){"use strict";var r=n(0),o=n.n(r);t.a=o.a.createContext(null)},,function(e,t){e.exports=function(e){return e&&e.__esModule?e:{default:e}}},function(e,t,n){"use strict";t.a=function(e){var t={};return function(n){return void 0===t[n]&&(t[n]=e(n)),t[n]}}},function(e,t,n){"use strict";function r(e){var t,n=arguments.length>1&&void 0!==arguments[1]?arguments[1]:166;function r(){for(var r=arguments.length,o=new Array(r),a=0;a=0||(o[n]=e[n]);return o}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";function r(e){if(Symbol.iterator in Object(e)||"[object Arguments]"===Object.prototype.toString.call(e))return Array.from(e)}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";function r(e,t){if(null==e)return{};var n,r,o={},a=Object.keys(e);for(r=0;r=0||(o[n]=e[n]);return o}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";function r(e,t){e.prototype=Object.create(t.prototype),e.prototype.constructor=e,e.__proto__=t}n.d(t,"a",function(){return r})},function(e,t,n){"use strict";var r=n(98),o={childContextTypes:!0,contextType:!0,contextTypes:!0,defaultProps:!0,displayName:!0,getDefaultProps:!0,getDerivedStateFromError:!0,getDerivedStateFromProps:!0,mixins:!0,propTypes:!0,type:!0},a={name:!0,length:!0,prototype:!0,caller:!0,callee:!0,arguments:!0,arity:!0},i={$$typeof:!0,compare:!0,defaultProps:!0,displayName:!0,propTypes:!0,type:!0},l={};function u(e){return r.isMemo(e)?i:l[e.$$typeof]||o}l[r.ForwardRef]={$$typeof:!0,render:!0,defaultProps:!0,displayName:!0,propTypes:!0};var s=Object.defineProperty,c=Object.getOwnPropertyNames,d=Object.getOwnPropertySymbols,f=Object.getOwnPropertyDescriptor,p=Object.getPrototypeOf,h=Object.prototype;e.exports=function e(t,n,r){if("string"!==typeof n){if(h){var o=p(n);o&&o!==h&&e(t,o,r)}var i=c(n);d&&(i=i.concat(d(n)));for(var l=u(t),m=u(n),v=0;v=0||(o[n]=e[n]);return o}(e,t);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);for(r=0;r=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(o[n]=e[n])}return o}n.d(t,"a",function(){return r})},,,function(e,t,n){"use strict";function r(e){return(r="function"===typeof Symbol&&"symbol"===typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"===typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e})(e)}function o(e){return(o="function"===typeof Symbol&&"symbol"===r(Symbol.iterator)?function(e){return r(e)}:function(e){return e&&"function"===typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":r(e)})(e)}n.d(t,"a",function(){return o})},function(e,t,n){"use strict";var r=n(31);Object.defineProperty(t,"__esModule",{value:!0}),t.default=void 0;var o=r(n(0)),a=(0,r(n(37)).default)(o.default.createElement(o.default.Fragment,null,o.default.createElement("path",{d:"M19 3H4.99c-1.11 0-1.98.9-1.98 2L3 19c0 1.1.88 2 1.99 2H19c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 12h-4c0 1.66-1.35 3-3 3s-3-1.34-3-3H4.99V5H19v10zm-3-5h-2V7h-4v3H8l4 4 4-4z"}),o.default.createElement("path",{fill:"none",d:"M0 0h24v24H0V0z"})),"MoveToInbox");t.default=a},function(e,t,n){"use strict";var r=n(31);Object.defineProperty(t,"__esModule",{value:!0}),t.default=void 0;var o=r(n(0)),a=(0,r(n(37)).default)(o.default.createElement("path",{d:"M20 4H4c-1.1 0-1.99.9-1.99 2L2 18c0 1.1.9 2 2 2h16c1.1 0 2-.9 2-2V6c0-1.1-.9-2-2-2zm0 4l-8 5-8-5V6l8 5 8-5v2z"}),"Mail");t.default=a},function(e,t,n){"use strict";var r=n(31);Object.defineProperty(t,"__esModule",{value:!0}),t.default=void 0;var o=r(n(0)),a=(0,r(n(37)).default)(o.default.createElement("path",{d:"M16.59 8.59L12 13.17 7.41 8.59 6 10l6 6 6-6z"}),"ExpandMore");t.default=a},,,,function(e,t,n){"use strict";n.r(t);var r=n(72);n.d(t,"default",function(){return r.a})},function(e,t,n){"use strict";var r=n(1),o=n(2),a=n(0),i=n.n(a),l=(n(5),n(3)),u=n(4),s=n(7),c=i.a.forwardRef(function(e,t){var n=e.children,a=e.classes,u=e.className,c=e.color,d=void 0===c?"inherit":c,f=e.component,p=void 0===f?"svg":f,h=e.fontSize,m=void 0===h?"default":h,v=e.htmlColor,b=e.titleAccess,g=e.viewBox,y=void 0===g?"0 0 24 24":g,x=Object(o.a)(e,["children","classes","className","color","component","fontSize","htmlColor","titleAccess","viewBox"]);return i.a.createElement(p,Object(r.a)({className:Object(l.a)(a.root,u,"inherit"!==d&&a["color".concat(Object(s.a)(d))],"default"!==m&&a["fontSize".concat(Object(s.a)(m))]),focusable:"false",viewBox:y,color:v,"aria-hidden":b?"false":"true",role:b?"img":"presentation",ref:t},x),n,b?i.a.createElement("title",null,b):null)});c.muiName="SvgIcon",t.a=Object(u.a)(function(e){return{root:{userSelect:"none",width:"1em",height:"1em",display:"inline-block",fill:"currentColor",flexShrink:0,fontSize:e.typography.pxToRem(24),transition:e.transitions.create("fill",{duration:e.transitions.duration.shorter})},colorPrimary:{color:e.palette.primary.main},colorSecondary:{color:e.palette.secondary.main},colorAction:{color:e.palette.action.active},colorError:{color:e.palette.error.main},colorDisabled:{color:e.palette.action.disabled},fontSizeInherit:{fontSize:"inherit"},fontSizeSmall:{fontSize:e.typography.pxToRem(20)},fontSizeLarge:{fontSize:e.typography.pxToRem(35)}}},{name:"MuiSvgIcon"})(c)},function(e,t,n){var r=n(93);e.exports=function(e){for(var t=1;t0&&void 0!==arguments[0]?arguments[0]:{};return Object(o.a)({paddingLeft:t(2),paddingRight:t(2)},n,Object(r.a)({},e.up("sm"),Object(o.a)({paddingLeft:t(3),paddingRight:t(3)},n[e.up("sm")])))},toolbar:(a={minHeight:56},Object(r.a)(a,"".concat(e.up("xs")," and (orientation: landscape)"),{minHeight:48}),Object(r.a)(a,e.up("sm"),{minHeight:64}),a)},n)}var f={50:"#e8eaf6",100:"#c5cae9",200:"#9fa8da",300:"#7986cb",400:"#5c6bc0",500:"#3f51b5",600:"#3949ab",700:"#303f9f",800:"#283593",900:"#1a237e",A100:"#8c9eff",A200:"#536dfe",A400:"#3d5afe",A700:"#304ffe"},p={50:"#fce4ec",100:"#f8bbd0",200:"#f48fb1",300:"#f06292",400:"#ec407a",500:"#e91e63",600:"#d81b60",700:"#c2185b",800:"#ad1457",900:"#880e4f",A100:"#ff80ab",A200:"#ff4081",A400:"#f50057",A700:"#c51162"},h={50:"#fafafa",100:"#f5f5f5",200:"#eeeeee",300:"#e0e0e0",400:"#bdbdbd",500:"#9e9e9e",600:"#757575",700:"#616161",800:"#424242",900:"#212121",A100:"#d5d5d5",A200:"#aaaaaa",A400:"#303030",A700:"#616161"},m={50:"#ffebee",100:"#ffcdd2",200:"#ef9a9a",300:"#e57373",400:"#ef5350",500:"#f44336",600:"#e53935",700:"#d32f2f",800:"#c62828",900:"#b71c1c",A100:"#ff8a80",A200:"#ff5252",A400:"#ff1744",A700:"#d50000"},v={black:"#000",white:"#fff"},b=n(11),g={text:{primary:"rgba(0, 0, 0, 0.87)",secondary:"rgba(0, 0, 0, 0.54)",disabled:"rgba(0, 0, 0, 0.38)",hint:"rgba(0, 0, 0, 0.38)"},divider:"rgba(0, 0, 0, 0.12)",background:{paper:v.white,default:h[50]},action:{active:"rgba(0, 0, 0, 0.54)",hover:"rgba(0, 0, 0, 0.08)",hoverOpacity:.08,selected:"rgba(0, 0, 0, 0.14)",disabled:"rgba(0, 0, 0, 0.26)",disabledBackground:"rgba(0, 0, 0, 0.12)"}},y={text:{primary:v.white,secondary:"rgba(255, 255, 255, 0.7)",disabled:"rgba(255, 255, 255, 0.5)",hint:"rgba(255, 255, 255, 0.5)",icon:"rgba(255, 255, 255, 0.5)"},divider:"rgba(255, 255, 255, 0.12)",background:{paper:h[800],default:"#303030"},action:{active:v.white,hover:"rgba(255, 255, 255, 0.1)",hoverOpacity:.1,selected:"rgba(255, 255, 255, 0.2)",disabled:"rgba(255, 255, 255, 0.3)",disabledBackground:"rgba(255, 255, 255, 0.12)"}};function x(e,t,n,r){e[t]||(e.hasOwnProperty(n)?e[t]=e[n]:"light"===t?e.light=Object(b.d)(e.main,r):"dark"===t&&(e.dark=Object(b.a)(e.main,1.5*r)))}function k(e){return Math.round(1e5*e)/1e5}var w={textTransform:"uppercase"},O='"Roboto", "Helvetica", "Arial", sans-serif';function S(e,t){var n="function"===typeof t?t(e):t,r=n.fontFamily,i=void 0===r?O:r,u=n.fontSize,s=void 0===u?14:u,c=n.fontWeightLight,d=void 0===c?300:c,f=n.fontWeightRegular,p=void 0===f?400:f,h=n.fontWeightMedium,m=void 0===h?500:h,v=n.fontWeightBold,b=void 0===v?700:v,g=n.htmlFontSize,y=void 0===g?16:g,x=n.allVariants,S=Object(a.a)(n,["fontFamily","fontSize","fontWeightLight","fontWeightRegular","fontWeightMedium","fontWeightBold","htmlFontSize","allVariants"]),C=s/14,E=function(e){return"".concat(e/y*C,"rem")},j=function(e,t,n,r,a){return Object(o.a)({fontFamily:i,fontWeight:e,fontSize:E(t),lineHeight:n},i===O?{letterSpacing:"".concat(k(r/t),"em")}:{},{},a,{},x)},T={h1:j(d,96,1,-1.5),h2:j(d,60,1,-.5),h3:j(p,48,1.04,0),h4:j(p,34,1.17,.25),h5:j(p,24,1.33,0),h6:j(m,20,1.6,.15),subtitle1:j(p,16,1.75,.15),subtitle2:j(m,14,1.57,.1),body1:j(p,16,1.5,.15),body2:j(p,14,1.43,.15),button:j(m,14,1.75,.4,w),caption:j(p,12,1.66,.4),overline:j(p,12,2.66,1,w)};return l()(Object(o.a)({htmlFontSize:y,pxToRem:E,round:k,fontFamily:i,fontSize:s,fontWeightLight:d,fontWeightRegular:p,fontWeightMedium:m,fontWeightBold:b},T),S,{clone:!1})}var C=.2,E=.14,j=.12;function T(){return["".concat(arguments.length<=0?void 0:arguments[0],"px ").concat(arguments.length<=1?void 0:arguments[1],"px ").concat(arguments.length<=2?void 0:arguments[2],"px ").concat(arguments.length<=3?void 0:arguments[3],"px rgba(0,0,0,").concat(C,")"),"".concat(arguments.length<=4?void 0:arguments[4],"px ").concat(arguments.length<=5?void 0:arguments[5],"px ").concat(arguments.length<=6?void 0:arguments[6],"px ").concat(arguments.length<=7?void 0:arguments[7],"px rgba(0,0,0,").concat(E,")"),"".concat(arguments.length<=8?void 0:arguments[8],"px ").concat(arguments.length<=9?void 0:arguments[9],"px ").concat(arguments.length<=10?void 0:arguments[10],"px ").concat(arguments.length<=11?void 0:arguments[11],"px rgba(0,0,0,").concat(j,")")].join(",")}var P=["none",T(0,1,3,0,0,1,1,0,0,2,1,-1),T(0,1,5,0,0,2,2,0,0,3,1,-2),T(0,1,8,0,0,3,4,0,0,3,3,-2),T(0,2,4,-1,0,4,5,0,0,1,10,0),T(0,3,5,-1,0,5,8,0,0,1,14,0),T(0,3,5,-1,0,6,10,0,0,1,18,0),T(0,4,5,-2,0,7,10,1,0,2,16,1),T(0,5,5,-3,0,8,10,1,0,3,14,2),T(0,5,6,-3,0,9,12,1,0,3,16,2),T(0,6,6,-3,0,10,14,1,0,4,18,3),T(0,6,7,-4,0,11,15,1,0,4,20,3),T(0,7,8,-4,0,12,17,2,0,5,22,4),T(0,7,8,-4,0,13,19,2,0,5,24,4),T(0,7,9,-4,0,14,21,2,0,5,26,4),T(0,8,9,-5,0,15,22,2,0,6,28,5),T(0,8,10,-5,0,16,24,2,0,6,30,5),T(0,8,11,-5,0,17,26,2,0,6,32,5),T(0,9,11,-5,0,18,28,2,0,7,34,6),T(0,9,12,-6,0,19,29,2,0,7,36,6),T(0,10,13,-6,0,20,31,3,0,8,38,7),T(0,10,13,-6,0,21,33,3,0,8,40,7),T(0,10,14,-6,0,22,35,3,0,8,42,7),T(0,11,14,-7,0,23,36,3,0,9,44,8),T(0,11,15,-7,0,24,38,3,0,9,46,8)],R={borderRadius:4};var N=n(18),_=n(54);t.a=function(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=e.breakpoints,n=void 0===t?{}:t,r=e.mixins,i=void 0===r?{}:r,u=e.palette,k=void 0===u?{}:u,w=e.shadows,O=e.spacing,C=e.typography,E=void 0===C?{}:C,j=Object(a.a)(e,["breakpoints","mixins","palette","shadows","spacing","typography"]),T=function(e){var t=e.primary,n=void 0===t?{light:f[300],main:f[500],dark:f[700]}:t,r=e.secondary,i=void 0===r?{light:p.A200,main:p.A400,dark:p.A700}:r,u=e.error,s=void 0===u?{light:m[300],main:m[500],dark:m[700]}:u,c=e.type,d=void 0===c?"light":c,k=e.contrastThreshold,w=void 0===k?3:k,O=e.tonalOffset,S=void 0===O?.2:O,C=Object(a.a)(e,["primary","secondary","error","type","contrastThreshold","tonalOffset"]);function E(e){return Object(b.c)(e,y.text.primary)>=w?y.text.primary:g.text.primary}function j(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:500,n=arguments.length>2&&void 0!==arguments[2]?arguments[2]:300,r=arguments.length>3&&void 0!==arguments[3]?arguments[3]:700;return!(e=Object(o.a)({},e)).main&&e[t]&&(e.main=e[t]),x(e,"light",n,S),x(e,"dark",r,S),e.contrastText||(e.contrastText=E(e.main)),e}var T={dark:y,light:g};return l()(Object(o.a)({common:v,type:d,primary:j(n),secondary:j(i,"A400","A200","A700"),error:j(s),grey:h,contrastThreshold:w,getContrastText:E,augmentColor:j,tonalOffset:S},T[d]),C,{clone:!1})}(k),A=function(e){var t=e.values,n=void 0===t?{xs:0,sm:600,md:960,lg:1280,xl:1920}:t,r=e.unit,i=void 0===r?"px":r,l=e.step,u=void 0===l?5:l,s=Object(a.a)(e,["values","unit","step"]);function d(e){var t="number"===typeof n[e]?n[e]:e;return"@media (min-width:".concat(t).concat(i,")")}function f(e,t){var r=c.indexOf(t)+1;return r===c.length?d(e):"@media (min-width:".concat(n[e]).concat(i,") and ")+"(max-width:".concat(n[c[r]]-u/100).concat(i,")")}return Object(o.a)({keys:c,values:n,up:d,down:function(e){var t=c.indexOf(e)+1,r=n[c[t]];return t===c.length?d("xs"):"@media (max-width:".concat(("number"===typeof r&&t>0?r:e)-u/100).concat(i,")")},between:f,only:function(e){return f(e,e)},width:function(e){return n[e]}},s)}(n),M=function(){var e,t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:8;if(t.mui)return t;e="function"===typeof t?t:function(e){return t*e};var n=function(){for(var t=arguments.length,n=new Array(t),r=0;r0&&void 0!==arguments[0]?arguments[0]:{},t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{},n=arguments.length>2?arguments[2]:void 0,r=t.pulsate,o=void 0!==r&&r,i=t.center,l=void 0===i?a||t.pulsate:i,u=t.fakeElement,s=void 0!==u&&u;if("mousedown"===e.type&&v.current)v.current=!1;else{"touchstart"===e.type&&(v.current=!0);var c,d,f,p=s?null:y.current,h=p?p.getBoundingClientRect():{width:0,height:0,left:0,top:0};if(l||0===e.clientX&&0===e.clientY||!e.clientX&&!e.touches)c=Math.round(h.width/2),d=Math.round(h.height/2);else{var m=e.clientX?e.clientX:e.touches[0].clientX,k=e.clientY?e.clientY:e.touches[0].clientY;c=Math.round(m-h.left),d=Math.round(k-h.top)}if(l)(f=Math.sqrt((2*Math.pow(h.width,2)+Math.pow(h.height,2))/3))%2===0&&(f+=1);else{var w=2*Math.max(Math.abs((p?p.clientWidth:0)-c),c)+2,O=2*Math.max(Math.abs((p?p.clientHeight:0)-d),d)+2;f=Math.sqrt(Math.pow(w,2)+Math.pow(O,2))}e.touches?(g.current=function(){x({pulsate:o,rippleX:c,rippleY:d,rippleSize:f,cb:n})},b.current=setTimeout(function(){g.current&&(g.current(),g.current=null)},80)):x({pulsate:o,rippleX:c,rippleY:d,rippleSize:f,cb:n})}},[a,x]),w=i.a.useCallback(function(){k({},{pulsate:!0})},[k]),O=i.a.useCallback(function(e,t){if(clearTimeout(b.current),"touchend"===e.type&&g.current)return e.persist(),g.current(),g.current=null,void(b.current=setTimeout(function(){O(e,t)}));g.current=null,p(function(e){return e.length>0?e.slice(1):e}),m.current=t},[]);return i.a.useImperativeHandle(t,function(){return{pulsate:w,start:k,stop:O}},[w,k,O]),i.a.createElement("span",Object(r.a)({className:Object(s.a)(l.root,u),ref:y},c),i.a.createElement(z,{component:null,exit:!0},f))});var W,B=Object(f.a)(function(e){return{root:{display:"block",position:"absolute",overflow:"hidden",borderRadius:"inherit",width:"100%",height:"100%",left:0,top:0,pointerEvents:"none",zIndex:0},ripple:{opacity:0,position:"absolute"},rippleVisible:{opacity:.3,transform:"scale(1)",animation:"$mui-ripple-enter ".concat(550,"ms ").concat(e.transitions.easing.easeInOut)},ripplePulsate:{animationDuration:"".concat(e.transitions.duration.shorter,"ms")},child:{opacity:1,display:"block",width:"100%",height:"100%",borderRadius:"50%",backgroundColor:"currentColor"},childLeaving:{opacity:0,animation:"$mui-ripple-exit ".concat(550,"ms ").concat(e.transitions.easing.easeInOut)},childPulsate:{position:"absolute",left:0,top:0,animation:"$mui-ripple-pulsate 2500ms ".concat(e.transitions.easing.easeInOut," 200ms infinite")},"@keyframes mui-ripple-enter":{"0%":{transform:"scale(0)",opacity:.1},"100%":{transform:"scale(1)",opacity:.3}},"@keyframes mui-ripple-exit":{"0%":{opacity:1},"100%":{opacity:0}},"@keyframes mui-ripple-pulsate":{"0%":{transform:"scale(1)"},"50%":{transform:"scale(0.92)"},"100%":{transform:"scale(1)"}}}},{flip:!1,name:"MuiTouchRipple"})(((W=i.a.memo(F)).muiName="MuiTouchRipple",W)),U=i.a.forwardRef(function(e,t){var n=e.action,a=e.buttonRef,l=e.centerRipple,f=void 0!==l&&l,p=e.children,m=e.classes,v=e.className,b=e.component,g=void 0===b?"button":b,y=e.disabled,x=e.disableRipple,k=void 0!==x&&x,w=e.disableTouchRipple,O=void 0!==w&&w,C=e.focusRipple,E=void 0!==C&&C,j=e.focusVisibleClassName,T=e.onBlur,P=e.onClick,R=e.onFocus,N=e.onFocusVisible,_=e.onKeyDown,A=e.onKeyUp,M=e.onMouseDown,I=e.onMouseLeave,z=e.onMouseUp,D=e.onTouchEnd,L=e.onTouchMove,F=e.onTouchStart,W=e.onDragLeave,U=e.tabIndex,H=void 0===U?0:U,$=e.TouchRippleProps,V=e.type,q=void 0===V?"button":V,K=Object(o.a)(e,["action","buttonRef","centerRipple","children","classes","className","component","disabled","disableRipple","disableTouchRipple","focusRipple","focusVisibleClassName","onBlur","onClick","onFocus","onFocusVisible","onKeyDown","onKeyUp","onMouseDown","onMouseLeave","onMouseUp","onTouchEnd","onTouchMove","onTouchStart","onDragLeave","tabIndex","TouchRippleProps","type"]),X=i.a.useRef(null);var G=i.a.useRef(null),Y=i.a.useState(!1),Q=Y[0],J=Y[1];y&&Q&&J(!1);var Z=S(),ee=Z.isFocusVisible,te=Z.onBlurVisible,ne=Z.ref;function re(e,t){var n=arguments.length>2&&void 0!==arguments[2]?arguments[2]:O;return Object(d.a)(function(r){return t&&t(r),!(r.defaultPrevented||n)&&G.current&&G.current[e](r),!0})}i.a.useImperativeHandle(n,function(){return{focusVisible:function(){J(!0),X.current.focus()}}},[]),i.a.useEffect(function(){Q&&E&&!k&&G.current.pulsate()},[k,E,Q]);var oe=re("start",M),ae=re("stop",W),ie=re("stop",z),le=re("stop",function(e){Q&&e.preventDefault(),I&&I(e)}),ue=re("start",F),se=re("stop",D),ce=re("stop",L),de=re("stop",function(e){Q&&(te(e),J(!1)),T&&T(e)},!1),fe=Object(d.a)(function(e){y||(X.current||(X.current=e.currentTarget),ee(e)&&(J(!0),N&&N(e)),R&&R(e))}),pe=i.a.useRef(!1),he=Object(d.a)(function(e){E&&!pe.current&&Q&&G.current&&" "===e.key&&(pe.current=!0,e.persist(),G.current.stop(e,function(){G.current.start(e)})),_&&_(e);var t=u.a.findDOMNode(X.current);e.target!==e.currentTarget||!g||"button"===g||" "!==e.key&&"Enter"!==e.key||"A"===t.tagName&&t.href||(e.preventDefault(),P&&P(e))}),me=Object(d.a)(function(e){E&&" "===e.key&&G.current&&Q&&(pe.current=!1,e.persist(),G.current.stop(e,function(){G.current.pulsate(e)})),A&&A(e)}),ve=Object(s.a)(m.root,v,Q&&[m.focusVisible,j],y&&m.disabled),be=g;"button"===be&&K.href&&(be="a");var ge={};"button"===be?(ge.type=q,ge.disabled=y):("a"===be&&K.href||(ge.role="button"),ge["aria-disabled"]=y);var ye=Object(c.c)(a,t),xe=Object(c.c)(ne,X),ke=Object(c.c)(ye,xe);return i.a.createElement(be,Object(r.a)({className:ve,onBlur:de,onClick:P,onFocus:fe,onKeyDown:he,onKeyUp:me,onMouseDown:oe,onMouseLeave:le,onMouseUp:ie,onDragLeave:ae,onTouchEnd:se,onTouchMove:ce,onTouchStart:ue,ref:ke,tabIndex:y?-1:H},ge,K),p,k||y?null:i.a.createElement(h,null,i.a.createElement(B,Object(r.a)({ref:G,center:f},$))))});t.a=Object(f.a)({root:{display:"inline-flex",alignItems:"center",justifyContent:"center",position:"relative",WebkitTapHighlightColor:"transparent",backgroundColor:"transparent",outline:0,border:0,margin:0,borderRadius:0,padding:0,cursor:"pointer",userSelect:"none",verticalAlign:"middle","-moz-appearance":"none","-webkit-appearance":"none",textDecoration:"none",color:"inherit","&::-moz-focus-inner":{borderStyle:"none"},"&$disabled":{pointerEvents:"none",cursor:"default"}},disabled:{},focusVisible:{}},{name:"MuiButtonBase"})(U)},,,,,,,,,function(e,t,n){"use strict";var r=n(17),o="function"===typeof Symbol&&Symbol.for,a=o?Symbol.for("react.element"):60103,i=o?Symbol.for("react.portal"):60106,l=o?Symbol.for("react.fragment"):60107,u=o?Symbol.for("react.strict_mode"):60108,s=o?Symbol.for("react.profiler"):60114,c=o?Symbol.for("react.provider"):60109,d=o?Symbol.for("react.context"):60110,f=o?Symbol.for("react.concurrent_mode"):60111,p=o?Symbol.for("react.forward_ref"):60112,h=o?Symbol.for("react.suspense"):60113,m=o?Symbol.for("react.memo"):60115,v=o?Symbol.for("react.lazy"):60116,b="function"===typeof Symbol&&Symbol.iterator;function g(e){for(var t=arguments.length-1,n="https://reactjs.org/docs/error-decoder.html?invariant="+e,r=0;r_.length&&_.push(e)}function I(e,t,n){return null==e?0:function e(t,n,r,o){var l=typeof t;"undefined"!==l&&"boolean"!==l||(t=null);var u=!1;if(null===t)u=!0;else switch(l){case"string":case"number":u=!0;break;case"object":switch(t.$$typeof){case a:case i:u=!0}}if(u)return r(o,t,""===n?"."+z(t,0):n),1;if(u=0,n=""===n?".":n+":",Array.isArray(t))for(var s=0;sthis.eventPool.length&&this.eventPool.push(e)}function de(e){e.eventPool=[],e.getPooled=se,e.release=ce}o(ue.prototype,{preventDefault:function(){this.defaultPrevented=!0;var e=this.nativeEvent;e&&(e.preventDefault?e.preventDefault():"unknown"!==typeof e.returnValue&&(e.returnValue=!1),this.isDefaultPrevented=ie)},stopPropagation:function(){var e=this.nativeEvent;e&&(e.stopPropagation?e.stopPropagation():"unknown"!==typeof e.cancelBubble&&(e.cancelBubble=!0),this.isPropagationStopped=ie)},persist:function(){this.isPersistent=ie},isPersistent:le,destructor:function(){var e,t=this.constructor.Interface;for(e in t)this[e]=null;this.nativeEvent=this._targetInst=this.dispatchConfig=null,this.isPropagationStopped=this.isDefaultPrevented=le,this._dispatchInstances=this._dispatchListeners=null}}),ue.Interface={type:null,target:null,currentTarget:function(){return null},eventPhase:null,bubbles:null,cancelable:null,timeStamp:function(e){return e.timeStamp||Date.now()},defaultPrevented:null,isTrusted:null},ue.extend=function(e){function t(){}function n(){return r.apply(this,arguments)}var r=this;t.prototype=r.prototype;var a=new t;return o(a,n.prototype),n.prototype=a,n.prototype.constructor=n,n.Interface=o({},r.Interface,e),n.extend=r.extend,de(n),n},de(ue);var fe=ue.extend({data:null}),pe=ue.extend({data:null}),he=[9,13,27,32],me=V&&"CompositionEvent"in window,ve=null;V&&"documentMode"in document&&(ve=document.documentMode);var be=V&&"TextEvent"in window&&!ve,ge=V&&(!me||ve&&8=ve),ye=String.fromCharCode(32),xe={beforeInput:{phasedRegistrationNames:{bubbled:"onBeforeInput",captured:"onBeforeInputCapture"},dependencies:["compositionend","keypress","textInput","paste"]},compositionEnd:{phasedRegistrationNames:{bubbled:"onCompositionEnd",captured:"onCompositionEndCapture"},dependencies:"blur compositionend keydown keypress keyup mousedown".split(" ")},compositionStart:{phasedRegistrationNames:{bubbled:"onCompositionStart",captured:"onCompositionStartCapture"},dependencies:"blur compositionstart keydown keypress keyup mousedown".split(" ")},compositionUpdate:{phasedRegistrationNames:{bubbled:"onCompositionUpdate",captured:"onCompositionUpdateCapture"},dependencies:"blur compositionupdate keydown keypress keyup mousedown".split(" ")}},ke=!1;function we(e,t){switch(e){case"keyup":return-1!==he.indexOf(t.keyCode);case"keydown":return 229!==t.keyCode;case"keypress":case"mousedown":case"blur":return!0;default:return!1}}function Oe(e){return"object"===typeof(e=e.detail)&&"data"in e?e.data:null}var Se=!1;var Ce={eventTypes:xe,extractEvents:function(e,t,n,r){var o=void 0,a=void 0;if(me)e:{switch(e){case"compositionstart":o=xe.compositionStart;break e;case"compositionend":o=xe.compositionEnd;break e;case"compositionupdate":o=xe.compositionUpdate;break e}o=void 0}else Se?we(e,n)&&(o=xe.compositionEnd):"keydown"===e&&229===n.keyCode&&(o=xe.compositionStart);return o?(ge&&"ko"!==n.locale&&(Se||o!==xe.compositionStart?o===xe.compositionEnd&&Se&&(a=ae()):(re="value"in(ne=r)?ne.value:ne.textContent,Se=!0)),o=fe.getPooled(o,t,n,r),a?o.data=a:null!==(a=Oe(n))&&(o.data=a),$(o),a=o):a=null,(e=be?function(e,t){switch(e){case"compositionend":return Oe(t);case"keypress":return 32!==t.which?null:(ke=!0,ye);case"textInput":return(e=t.data)===ye&&ke?null:e;default:return null}}(e,n):function(e,t){if(Se)return"compositionend"===e||!me&&we(e,t)?(e=ae(),oe=re=ne=null,Se=!1,e):null;switch(e){case"paste":return null;case"keypress":if(!(t.ctrlKey||t.altKey||t.metaKey)||t.ctrlKey&&t.altKey){if(t.char&&1