ss.sh 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. #!/bin/bash
  2. # Author: yeho <lj2007331 AT gmail.com>
  3. # BLOG: https://blog.linuxeye.com
  4. #
  5. # Notes: OneinStack for CentOS/RadHat 6+ Debian 7+ and Ubuntu 12+
  6. # Install SS Server
  7. #
  8. # Project home page:
  9. # https://oneinstack.com
  10. # https://github.com/lj2007331/oneinstack
  11. export PATH=/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin
  12. clear
  13. printf "
  14. #######################################################################
  15. # OneinStack for CentOS/RadHat 6+ Debian 6+ and Ubuntu 12+ #
  16. # Install SS Server #
  17. # For more information please visit https://oneinstack.com #
  18. #######################################################################
  19. "
  20. # get pwd
  21. sed -i "s@^oneinstack_dir.*@oneinstack_dir=$(pwd)@" ./options.conf
  22. pushd src > /dev/null
  23. . ../options.conf
  24. . ../versions.txt
  25. . ../include/color.sh
  26. . ../include/check_os.sh
  27. . ../include/download.sh
  28. . ../include/python.sh
  29. # Check if user is root
  30. [ $(id -u) != '0' ] && { echo "${CFAILURE}Error: You must be root to run this script${CEND}"; exit 1; }
  31. PUBLIC_IPADDR=$(../include/get_public_ipaddr.py)
  32. [ "${CentOS_RHEL_version}" == '5' ] && { echo "${CWARNING}SS only support CentOS6,7 or Debian or Ubuntu! ${CEND}"; exit 1; }
  33. Check_SS() {
  34. [ -f /usr/local/bin/ss-server ] && SS_version=1
  35. [ -f ${python_install_dir}/bin/ssserver ] && SS_version=2
  36. }
  37. AddUser_SS() {
  38. while :; do echo
  39. read -p "Please input password for SS: " SS_password
  40. [ -n "$(echo ${SS_password} | grep '[+|&]')" ] && { echo "${CWARNING}input error,not contain a plus sign (+) and & ${CEND}"; continue; }
  41. (( ${#SS_password} >= 5 )) && break || echo "${CWARNING}SS password least 5 characters! ${CEND}"
  42. done
  43. }
  44. Iptables_set() {
  45. if [ -e '/etc/sysconfig/iptables' ]; then
  46. SS_Already_port=$(grep -oE '9[0-9][0-9][0-9]' /etc/sysconfig/iptables | head -n 1)
  47. elif [ -e '/etc/iptables.up.rules' ]; then
  48. SS_Already_port=$(grep -oE '9[0-9][0-9][0-9]' /etc/iptables.up.rules | head -n 1)
  49. fi
  50. if [ -n "${SS_Already_port}" ]; then
  51. let SS_Default_port="${SS_Already_port}+1"
  52. else
  53. SS_Default_port=9001
  54. fi
  55. while :; do echo
  56. read -p "Please input SS port(Default: ${SS_Default_port}): " SS_port
  57. [ -z "${SS_port}" ] && SS_port=${SS_Default_port}
  58. if [ ${SS_port} -ge 1 >/dev/null 2>&1 -a ${SS_port} -le 65535 >/dev/null 2>&1 ]; then
  59. [ -z "$(netstat -tpln | grep :${SS_port}$)" ] && break || echo "${CWARNING}This port is already used! ${CEND}"
  60. else
  61. echo "${CWARNING}input error! Input range: 1~65535${CEND}"
  62. fi
  63. done
  64. if [ "${OS}" == 'CentOS' ]; then
  65. if [ -n "`grep 'dport 80 ' /etc/sysconfig/iptables`" -a -z "$(grep -E ${SS_port} /etc/sysconfig/iptables)" ]; then
  66. iptables -I INPUT 4 -p udp -m state --state NEW -m udp --dport ${SS_port} -j ACCEPT
  67. iptables -I INPUT 4 -p tcp -m state --state NEW -m tcp --dport ${SS_port} -j ACCEPT
  68. service iptables save
  69. fi
  70. elif [[ ${OS} =~ ^Ubuntu$|^Debian$ ]]; then
  71. if [ -n "`grep 'dport 80 ' /etc/iptables.up.rules`" -a -z "$(grep -E ${SS_port} /etc/iptables.up.rules)" ]; then
  72. iptables -I INPUT 4 -p udp -m state --state NEW -m udp --dport ${SS_port} -j ACCEPT
  73. iptables -I INPUT 4 -p tcp -m state --state NEW -m tcp --dport ${SS_port} -j ACCEPT
  74. iptables-save > /etc/iptables.up.rules
  75. fi
  76. fi
  77. }
  78. Def_parameter() {
  79. while :; do echo
  80. echo "Please select SS server version:"
  81. echo -e "\t${CMSG}1${CEND}. Install SS-libev"
  82. echo -e "\t${CMSG}2${CEND}. Install SS-python"
  83. read -p "Please input a number:(Default 1 press Enter) " SS_version
  84. [ -z "${SS_version}" ] && SS_version=1
  85. if [[ ! "${SS_version}" =~ ^[1-2]$ ]]; then
  86. echo "${CWARNING}input error! Please only input number 1~2${CEND}"
  87. else
  88. break
  89. fi
  90. done
  91. AddUser_SS
  92. Iptables_set
  93. if [ "${OS}" == "CentOS" ]; then
  94. pkgList="wget unzip openssl-devel gcc swig autoconf libtool libevent automake make curl curl-devel zlib-devel perl perl-devel cpio expat-devel gettext-devel git asciidoc xmlto c-ares-devel pcre-devel mbedtls-devel udns-devel libev-devel libsodium"
  95. for Package in ${pkgList}; do
  96. yum -y install ${Package}
  97. done
  98. elif [[ "${OS}" =~ ^Ubuntu$|^Debian$ ]]; then
  99. apt-get -y update
  100. pkgList="curl wget unzip gcc swig automake make perl cpio git libmbedtls-dev libudns-dev libev-dev gettext build-essential autoconf libtool libpcre3-dev asciidoc xmlto libc-ares-dev libsodium-dev"
  101. for Package in ${pkgList}; do
  102. apt-get -y install $Package
  103. done
  104. fi
  105. }
  106. Install_SS-python() {
  107. [ ! -e "${python_install_dir}/bin/python" ] && Install_Python
  108. ${python_install_dir}/bin/pip install M2Crypto
  109. ${python_install_dir}/bin/pip install greenlet
  110. ${python_install_dir}/bin/pip install gevent
  111. ${python_install_dir}/bin/pip install shadowsocks
  112. if [ -f ${python_install_dir}/bin/ssserver ]; then
  113. /bin/cp ../init.d/SS-python-init /etc/init.d/shadowsocks
  114. chmod +x /etc/init.d/shadowsocks
  115. sed -i "s@SS_bin=.*@SS_bin=${python_install_dir}/bin/ssserver@" /etc/init.d/shadowsocks
  116. [ "${OS}" == "CentOS" ] && { chkconfig --add shadowsocks; chkconfig shadowsocks on; }
  117. [[ "${OS}" =~ ^Ubuntu$|^Debian$ ]] && update-rc.d shadowsocks defaults
  118. else
  119. echo
  120. echo "${CQUESTION}SS-python install failed! Please visit https://oneinstack.com${CEND}"
  121. exit 1
  122. fi
  123. }
  124. Install_SS-libev() {
  125. src_url=http://mirrors.linuxeye.com/oneinstack/src/shadowsocks-libev-3.1.1.tar.gz && Download_src
  126. src_url=http://mirrors.linuxeye.com/oneinstack/src/libsodium-1.0.16.tar.gz && Download_src
  127. src_url=http://mirrors.linuxeye.com/oneinstack/src/mbedtls-2.6.0-gpl.tgz && Download_src
  128. tar xzf shadowsocks-libev-3.1.1.tar.gz
  129. tar xzf libsodium-1.0.16.tar.gz
  130. tar xzf mbedtls-2.6.0-gpl.tgz
  131. pushd libsodium-1.0.16
  132. ./configure
  133. make -j ${THREAD} && make install
  134. popd
  135. pushd mbedtls-2.6.0
  136. make SHARED=1 CFLAGS=-fPIC
  137. make DESTDIR=/usr install
  138. popd
  139. pushd shadowsocks-libev-3.1.1
  140. ./configure
  141. make -j ${THREAD} && make install
  142. popd
  143. echo '/usr/local/lib' > /etc/ld.so.conf.d/local.conf
  144. ldconfig
  145. if [ -f /usr/local/bin/ss-server ]; then
  146. if [ "${OS}" == "CentOS" ]; then
  147. /bin/cp ../init.d/SS-libev-init-CentOS /etc/init.d/shadowsocks
  148. chkconfig --add shadowsocks
  149. chkconfig shadowsocks on
  150. elif [[ "${OS}" =~ ^Ubuntu$|^Debian$ ]];then
  151. /bin/cp ../init.d/SS-libev-init-Ubuntu /etc/init.d/shadowsocks
  152. update-rc.d shadowsocks defaults
  153. fi
  154. else
  155. echo
  156. echo "${CQUESTION}SS-libev install failed! Please visit https://oneinstack.com${CEND}"
  157. exit 1
  158. fi
  159. }
  160. Uninstall_SS() {
  161. while :; do echo
  162. read -p "Do you want to uninstall SS? [y/n]: " SS_yn
  163. if [[ ! "${SS_yn}" =~ ^[y,n]$ ]]; then
  164. echo "${CWARNING}input error! Please only input 'y' or 'n'${CEND}"
  165. else
  166. break
  167. fi
  168. done
  169. if [ "${SS_yn}" == 'y' ]; then
  170. [ -n "$(ps -ef | grep -v grep | grep -iE "ssserver|ss-server")" ] && /etc/init.d/shadowsocks stop
  171. [ "${OS}" == "CentOS" ] && chkconfig --del shadowsocks
  172. [[ "${OS}" =~ ^Ubuntu$|^Debian$ ]] && update-rc.d -f shadowsocks remove
  173. rm -rf /etc/shadowsocks /var/run/shadowsocks.pid /etc/init.d/shadowsocks
  174. if [ "${SS_version}" == '1' ]; then
  175. rm -f /usr/local/bin/{ss-local,ss-tunnel,ss-server,ss-manager,ss-redir}
  176. rm -f /usr/local/lib/libshadowsocks.*
  177. rm -f /usr/local/include/shadowsocks.h
  178. rm -f /usr/local/lib/pkgconfig/shadowsocks-libev.pc
  179. rm -f /usr/local/share/man/man1/{ss-local.1,ss-tunnel.1,ss-server.1,ss-manager.1,ss-redir.1,shadowsocks.8}
  180. if [ $? -eq 0 ]; then
  181. echo "${CSUCCESS}SS-libev uninstall successful! ${CEND}"
  182. else
  183. echo "${CFAILURE}SS-libev uninstall failed! ${CEND}"
  184. fi
  185. elif [ "${SS_version}" == '2' ]; then
  186. ${python_install_dir}/bin/pip uninstall -y shadowsocks
  187. if [ $? -eq 0 ]; then
  188. echo "${CSUCCESS}SS-python uninstall successful! ${CEND}"
  189. else
  190. echo "${CFAILURE}SS-python uninstall failed! ${CEND}"
  191. fi
  192. fi
  193. fi
  194. }
  195. Config_SS() {
  196. [ ! -d "/etc/shadowsocks" ] && mkdir /etc/shadowsocks
  197. [ "${SS_version}" == '1' ] && cat > /etc/shadowsocks/config.json << EOF
  198. {
  199. "server":"0.0.0.0",
  200. "server_port":${SS_port},
  201. "local_address":"127.0.0.1",
  202. "local_port":1080,
  203. "password":"${SS_password}",
  204. "timeout":300,
  205. "method":"aes-256-cfb",
  206. }
  207. EOF
  208. [ "${SS_version}" == '2' ] && cat > /etc/shadowsocks/config.json << EOF
  209. {
  210. "server":"0.0.0.0",
  211. "local_address":"127.0.0.1",
  212. "local_port":1080,
  213. "port_password":{
  214. "${SS_port}":"${SS_password}"
  215. },
  216. "timeout":300,
  217. "method":"aes-256-cfb",
  218. "fast_open":false
  219. }
  220. EOF
  221. }
  222. AddUser_Config_SS() {
  223. [ ! -e /etc/shadowsocks/config.json ] && { echo "${CFAILURE}SS is not installed! ${CEND}"; exit 1; }
  224. [ -z "$(grep \"${SS_port}\" /etc/shadowsocks/config.json)" ] && sed -i "s@\"port_password\":{@\"port_password\":{\n\t\"${SS_port}\":\"${SS_password}\",@" /etc/shadowsocks/config.json || { echo "${CWARNING}This port is already in /etc/shadowsocks/config.json${CEND}"; exit 1; }
  225. }
  226. Print_User_SS() {
  227. printf "
  228. Your Server IP: ${CMSG}${PUBLIC_IPADDR}${CEND}
  229. Your Server Port: ${CMSG}${SS_port}${CEND}
  230. Your Password: ${CMSG}${SS_password}${CEND}
  231. Your Local IP: ${CMSG}127.0.0.1${CEND}
  232. Your Local Port: ${CMSG}1080${CEND}
  233. Your Encryption Method: ${CMSG}aes-256-cfb${CEND}
  234. "
  235. }
  236. case "$1" in
  237. install)
  238. Def_parameter
  239. [ "${SS_version}" == '1' ] && Install_SS-libev
  240. [ "${SS_version}" == '2' ] && Install_SS-python
  241. Config_SS
  242. service shadowsocks start
  243. Print_User_SS
  244. ;;
  245. adduser)
  246. Check_SS
  247. if [ "${SS_version}" == '2' ]; then
  248. AddUser_SS
  249. Iptables_set
  250. AddUser_Config_SS
  251. service shadowsocks restart
  252. Print_User_SS
  253. else
  254. printf "
  255. Sorry, we have no plan to support multi port configuration. Actually you can use multiple instances instead. For example:
  256. ss-server -c /etc/shadowsocks/config1.json -f /var/run/shadowsocks-server/pid1
  257. ss-server -c /etc/shadowsocks/config2.json -f /var/run/shadowsocks-server/pid2
  258. "
  259. fi
  260. ;;
  261. uninstall)
  262. Check_SS
  263. Uninstall_SS
  264. ;;
  265. *)
  266. echo
  267. echo "Usage: ${CMSG}$0${CEND} { ${CMSG}install${CEND} | ${CMSG}adduser${CEND} | ${CMSG}uninstall${CEND} }"
  268. echo
  269. exit 1
  270. esac