| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408 | 
							- #!/bin/bash
 
- _WWW='/www'
 
- _PWDFILE="${_WWW}/lib/htpasswd"
 
- _TMP="${_WWW}/tmp"
 
- _LOG="${_WWW}/log/admin.log"
 
- _DEBUG=1
 
- ## logging
 
- logThis() {
 
-   [[ $_DEBUG -gt 0 ]] || return 0
 
-   [[ $_ERR -gt 0 ]] && _TYPE='E:' || _TYPE='I:'  ## Info or Error indication
 
-   local _TIME=$(printf '%(%d.%m.%Y %H:%M:%S)T' -1)
 
-   printf '%b\n' "$_TIME  $_TYPE ${1} " >> $_LOG
 
-   [[ $_DEBUG -gt 1 ]] && printf '%b\n' "[verbose] $_TYPE ${1}"
 
-   return 0
 
- }
 
- ## http response
 
- headerPrint() {
 
-   case ${1} in
 
-     200) printf '%b' 'HTTP/1.1 200 OK\nAccess-Control-Allow-Origin: *\n\n';;
 
-     301) printf '%b' "HTTP/1.1 301 Moved Permanently\nLocation: ${HTTP_REFERER}\n\n";;
 
-     403) printf '%b' 'HTTP/1.1 403 Forbidden\n\n';;
 
-     405) printf '%b' 'HTTP/1.1 405 Method Not Allowed\n\n';;
 
-     406) printf '%b' 'HTTP/1.1 406 Not Acceptable\n\n';;
 
-   esac
 
-   return 0
 
- }
 
- htDigest() {
 
-   _USER='admin'
 
-   _PWD=$1
 
-   _REALM='superglue'
 
-   _HASH=$(echo -n "$_USER:$_REALM:$_PWD" | md5sum | cut -b -32)
 
-   echo -n "$_USER:$_REALM:$_HASH"
 
- }
 
- urlDec() {
 
-   local value=${*//+/%20}
 
-   for part in ${value//%/ \\x}; do
 
-     printf "%b%s" "${part:0:4}" "${part:4}"
 
-   done
 
- }
 
- setQueryVars() {
 
-   local _POST=$(cat)
 
-   local vars=${_POST//\*/%2A}
 
-   for var in ${vars//&/ }; do
 
-     local value=$(urlDec "${var#*=}")
 
-     value=${value//\\/\\\\}
 
-     eval "_${var%=*}=\"${value//\"/\\\"}\""
 
-   done
 
- }
 
- getQueryFile() {
 
-   _POST_TMP=$(mktemp -p $_TMP)  ## make tmp POST file
 
-   cat > $_POST_TMP  ## cautiously storing entire POST in a file
 
-   logThis "'multipart': decoding stream"
 
-   local _BND=$(findPostOpt 'boundary')
 
-   ## bash is binary unsafe and eats away precious lines
 
-   ## thus using gawk
 
-   function cutFile() {
 
-     gawk -v "want=$1" -v "bnd=$_BND" '
 
-       BEGIN { RS="\r\n"; ORS="\r\n" }
 
-       # reset based on boundaries
 
-       $0 == "--"bnd""     { st=1; next; }
 
-       $0 == "--"bnd"--"   { st=0; next; }
 
-       $0 == "--"bnd"--\r" { st=0; next; }
 
-       # search for wanted file
 
-       st == 1 && $0 ~  "^Content-Disposition:.* name=\""want"\"" { st=2; next; }
 
-       st == 1 && $0 == "" { st=9; next; }
 
-       # wait for newline, then start printing
 
-       st == 2 && $0 == "" { st=3; next; }
 
-       st == 3 { print $0 }
 
-       ' 2>&1
 
-   }
 
-   cutFile 'fwupload' < "${_POST_TMP}" > "${_TMP}/fwupload.bin"
 
- }
 
- ## find arbitrary option supplied in Content-Type header
 
- ## eg: "Content-Type:application/octet-stream; verbose=1"
 
- findPostOpt() {
 
-   for i in "${CONTENT_TYPE[@]:1}"; do
 
-     case "${i/=*}" in 
 
-       "$1") printf '%b' "${i/*=}" ;;
 
-     esac
 
-   done
 
-   return 0
 
- }
 
- runSuid() {
 
-   local _SID=$(/usr/bin/ps -p $$ -o sid=)  ## pass session id to the child
 
-   local _CMD=$@
 
-   sudo ./suid.sh $_CMD $_SID 2>&1
 
- }
 
- validIp() {
 
-   local _IP=$1
 
-   local _RET=1
 
-   if [[ $_IP =~ ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$ ]]; then
 
-     OIFS=$IFS
 
-     IFS='.'
 
-     _IP=($_IP)
 
-     IFS=$OIFS
 
-     [[ ${_IP[0]} -le 255 && ${_IP[1]} -le 255 && ${_IP[2]} -le 255 && ${_IP[3]} -le 255 ]]
 
-     _RET=$?
 
-   fi
 
-   return $_RET
 
- }
 
- pwdChange() {
 
-   if [[ ! -z "${_pwd##$_pwdd}" ]]; then 
 
-     _ERR=1
 
-     showMesg 'Passwords did not match'
 
-   fi
 
-   if [[ ${#_pwd} -lt 6 ]]; then
 
-     _ERR=1
 
-     showMesg 'Password must be at least 6 characters long'
 
-   fi
 
-   runSuid "echo -e \"$_pwd\n$_pwd\" | passwd root"
 
-   runSuid "echo $(htDigest $_pwd) > $_PWDFILE"
 
-   _ERR=$?
 
-   if [[ $_ERR -gt 0 ]]; then
 
-     showMesg 'Password change failed'
 
-   else
 
-     showMesg 'Password is changed'
 
-   fi
 
- }
 
- lanAddr() {
 
-   logThis "new LAN addr is: $_laddr"
 
-   validIp $_laddr || showMesg 'Not valid network address'
 
-   doUci set laddr $_laddr
 
-   _ERR=$?
 
-   if [[ $_ERR -gt 0 ]]; then
 
-     showMesg 'Setting network address failed'
 
-   else
 
-     (sleep 1; doUci commit network; doUci commit wireless;)&
 
-     showMesg 'New network address is set' "Your server is now accessible under <a href='http://$_laddr/admin'>http://$_laddr/admin</a>"
 
-   fi 
 
- }
 
- ssidChange() {
 
-   ## default enc for now
 
-   local _enc='psk2'
 
-   logThis "new ssid is: $_ssid"
 
-   logThis "new key is: $_key"
 
-   if [[ ${#_ssid} -lt 4 ]]; then
 
-    _ERR=1
 
-    showMesg 'SSID must be at least 4 characters long'
 
-   fi
 
-   doUci set ssid $_ssid
 
-   _ERR=$?
 
-   [[ $_ERR -gt 0 ]] && showMesg 'New SSID is not set'
 
-   if [[ -z $_key ]]; then
 
-     ## if key is empty set encryption to none and remove key
 
-     doUci set key && doUci set enc 'none'
 
-     _ERR=$?
 
-   else
 
-     if [[ ${#_key} -lt 8 ]]; then
 
-      _ERR=1
 
-       showMesg 'Passphrase must be at least 8 characters long'
 
-     fi
 
-     doUci set key $_key && doUci set enc $_enc
 
-     _ERR=$?
 
-     [[ $_ERR -gt 0 ]] && showMesg 'Passphrase is not set'
 
-   fi
 
-   [[ $_ERR -gt 0 ]] && showMesg 'Wireless changes failed'
 
-   doUci commit wireless && showMesg 'Wireless changes applied'
 
- }
 
- #showError() {
 
- #  headerPrint 406
 
- #  logThis "$@"
 
- #  echo "ERROR: $@"
 
- #  exit 1
 
- #}
 
- showMesg() {
 
-   logThis "$@"
 
-   local _MSG=$1
 
-   local _SUBMSG=$2
 
-   _MSG=${_MSG:='Not defined'}
 
-   _SUBMSG=${_SUBMSG:='back to control panel in a second..'}
 
-   if [[ $_ERR -gt 0 ]]; then
 
-     local _TYPE='ERROR: '
 
-     headerPrint 406
 
-   else
 
-     local _TYPE='OK: '
 
-     headerPrint 200
 
-   fi
 
-   htmlHead "<meta http-equiv='refresh' content='5;url=${HTTP_REFERER}'>"
 
-   echo "<body>
 
- <img src='/resources/img/superglueLogo.png' class='logo'>
 
- <hr>
 
- <h2 style='display:inline'>$_TYPE $_MSG</h2>
 
- <span style='display:inline; margin-left: 50px;'>$_SUBMSG</span>
 
- <hr>
 
- </body></html>"
 
-   exit 0
 
- }
 
- updateFw() {
 
-   logThis "updating fw"
 
-   _FWFILE="${_TMP}/fwupload.bin"
 
-   logThis "fwfile is: $(ls -lad $_FWFILE)"
 
-   _OUT="$(/sbin/sysupgrade -T $_FWFILE 2>&1)"
 
-   _ERR=$?
 
-   [[ $_ERR -gt 0 ]] && showMesg "$_OUT"
 
-   _OUT="$(runSuid /sbin/mtd -e firmware -q write $_FWFILE firmware)"
 
-   _ERR=$?
 
-   [[ $_ERR -gt 0 ]] && showMesg "mtd failed, $_OUT"
 
-   runSuid reboot
 
-   showMesg 'Firmware update is completed, rebooting..' 'this might take up to 60 seconds'
 
- }
 
- rebootNow() {
 
-   logThis "reboot: now!"
 
-   runSuid reboot
 
-   showMesg 'Rebooting..' 'this might take up to 60 seconds'
 
- }
 
- doUci() {
 
-   local _CMD=''
 
-   local _ARG=''
 
-   case $1 in
 
-     get|set|commit) _CMD=$1;;
 
-           *) logThis 'bad UCI command'; headerPrint 405; echo 'bad UCI command'; exit 1 ;;
 
-   esac
 
-   case $2 in
 
-     ssid) _ARG='wireless.@wifi-iface[0].ssid';;
 
-     enc) _ARG='wireless.@wifi-iface[0].encryption';;
 
-     key) _ARG='wireless.@wifi-iface[0].key';;
 
-     laddr) _ARG='network.lan.ipaddr';;
 
-     *) if [[ $_CMD == 'commit' ]]; then
 
-         _ARG=$2
 
-        else 
 
-         logThis 'bad UCI entry'
 
-         _ERR=1
 
-         showMesg 'bad UCI entry'
 
-        fi ;;
 
-   esac
 
-   if [[ $_CMD == 'get' ]]; then
 
-     if [ ! -z $_ARG ]; then
 
-       /sbin/uci get $_ARG
 
-     fi
 
-   fi
 
-   if [[ $_CMD == 'set' ]]; then
 
-     local _VAL=$3
 
-     if [ -z $_VAL ]; then
 
-       logThis "empty $_ARG value, removing record"
 
-       runSuid /sbin/uci delete $_ARG || ( echo "uci delete $_ARG: error"; exit 1; )
 
-     fi
 
-     if [ ! -z $_ARG ]; then
 
-       logThis "setting $_ARG value"
 
-       runSuid /sbin/uci set $_ARG=$_VAL || ( echo "uci set $_ARG: error"; exit 1; )
 
-     fi
 
-   fi
 
-   if [[ $_CMD == 'commit' ]]; then
 
-     runSuid /sbin/uci commit $_ARG|| echo "uci commit $_ARG: error"
 
-     if [[ "$_ARG" == 'wireless' ]]; then
 
-       runSuid /sbin/wifi || echo 'wifi: error'
 
-     fi
 
-     if [[ "$_ARG" == 'network' ]]; then
 
-       runSuid /etc/init.d/dnsmasq restart && runSuid /etc/init.d/network restart || echo 'network: error'
 
-     fi
 
-   fi
 
- }
 
- htmlHead() {
 
- echo "<!doctype html>
 
- <html>
 
- <head><title>SuperGlue | Administration</title>
 
- $@
 
- <style> 
 
- body { background:#ccc; color:#000; margin: 20px 0 0 200px; font-family: TitilliumWeb;}
 
- input { display: block; }
 
- .inline { display: inline; }
 
- img.logo { position: absolute; left:50px; top: 20px;}
 
- pre { white-space: pre-wrap; }
 
- @font-face { font-family: TitilliumWeb; src: url('/resources/fonts/Titillium_Web/TitilliumWeb-Regular.ttf') format('truetype'); }
 
- @font-face { font-family: TitilliumWeb; font-weight: bold; src: url('/resources/fonts/Titillium_Web/TitilliumWeb-Bold.ttf') format('truetype'); }
 
- </style>
 
- </head>"
 
- }
 
- ## unless auth is disabled in lighttpd
 
- ## it should never come to this, 
 
- if [[ -z $HTTP_AUTHORIZATION ]]; then
 
-   logThis 'no auth'
 
-   headerPrint 403
 
-   echo 'no is no'
 
-   exit 1
 
- else logThis 'auth OK'
 
- fi
 
- if [[ $REQUEST_METHOD == 'POST' ]]; then
 
-   if [[ $CONTENT_LENGTH -gt 0 ]]; then
 
-     CONTENT_TYPE=( ${CONTENT_TYPE} )
 
-     _CONTENT_TYPE="${CONTENT_TYPE[0]/;}"
 
-     _ENC="${HTTP_CONTENT_ENCODING}"
 
-     case "${_CONTENT_TYPE}" in
 
-       application/x-www-form-urlencoded) setQueryVars;;
 
-       multipart/form-data) getQueryFile;;
 
-       *) _ERR=1; _OUT='this is not a post' ;;
 
-     esac
 
-     case $REQUEST_URI in
 
-       /admin/pwdchange) pwdChange;;
 
-       /admin/ssidchange) ssidChange;;
 
-       /admin/lanaddr) lanAddr;;
 
-       /admin/updatefw) updateFw;;
 
-       /admin/rebootnow) rebootNow;;
 
-       *) logThis 'bad action'; headerPrint 405; echo 'no such thing'; exit 1;;
 
-     esac
 
-   fi
 
-   headerPrint 301
 
- fi
 
- headerPrint 200
 
- ## html head
 
- htmlHead
 
- echo "<body>
 
- <img src='/resources/img/superglueLogo.png' class='logo'>
 
- <hr>
 
- <h2 style='display:inline'>Superglue server control panel</h2>
 
- <span style='display:inline; margin-left: 50px;'>System version: "$(cat /etc/superglue_version || echo 'n/a')" | Device: "$(cat /etc/superglue_model || echo 'n/a')" | OpenWRT: "$(cat /etc/openwrt_version || echo 'n/a')"</span>
 
- <span style='display:block;'>$(uptime)</span>
 
- <hr>
 
- Change password:
 
- <form method='post' action='/admin/pwdchange'>
 
- <input type='text' name='usr' value='admin' readonly>
 
- <input type='password' name='pwd'>
 
- <input type='password' name='pwdd'>
 
- <input type='submit' value='Apply'>
 
- </form>
 
- <hr>
 
- Configure wireless network:
 
- <form method='post' action='/admin/ssidchange'>
 
- <input type='text' name='ssid' value='$(doUci get ssid)'>
 
- <input type='text' name='key' value='$(doUci get key)'>
 
- <input type='submit' value='Apply'>
 
- </form>
 
- <hr>
 
- Configure network address:
 
- <form method='post' action='/admin/lanaddr'>
 
- <input type='text' name='laddr' value='$(doUci get laddr)'>
 
- <input type='submit' value='Apply'>
 
- </form>
 
- <hr>
 
- Update firmware:
 
- <form method='post' action='/admin/updatefw' enctype='multipart/form-data'>
 
- <input type='file' name='fwupload'>
 
- <input type='submit' id='upload' value='Upload'>
 
- </form>
 
- <hr>
 
- <form action='/admin/rebootnow' method='post' class='inline'>
 
- <input type='hidden' name='reboot' value='now' class='inline'>
 
- <input type='submit' value='Reboot' class='inline'>
 
- </form>
 
- <form action='http://logout@${HTTP_HOST}/admin' method='get' class='inline'>
 
- <input type='submit' value='Logout' class='inline'>
 
- </form>
 
- <hr>
 
- Memory:
 
- <pre>$(free)</pre>
 
- <hr>
 
- Storage:
 
- <pre>$(df -h)</pre>
 
- <hr>
 
- Environment:
 
- <pre>$(env)</pre>
 
- <hr>
 
- $_POST
 
- </body></html>"
 
- exit 0
 
 
  |