freedombone-app-keyserver 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473
  1. #!/bin/bash
  2. #
  3. # .---. . .
  4. # | | |
  5. # |--- .--. .-. .-. .-.| .-. .--.--. |.-. .-. .--. .-.
  6. # | | (.-' (.-' ( | ( )| | | | )( )| | (.-'
  7. # ' ' --' --' -' - -' ' ' -' -' -' ' - --'
  8. #
  9. # Freedom in the Cloud
  10. #
  11. # SKS Keyserver
  12. #
  13. # License
  14. # =======
  15. #
  16. # Copyright (C) 2017 Bob Mottram <bob@freedombone.net>
  17. #
  18. # This program is free software: you can redistribute it and/or modify
  19. # it under the terms of the GNU Affero General Public License as published by
  20. # the Free Software Foundation, either version 3 of the License, or
  21. # (at your option) any later version.
  22. #
  23. # This program is distributed in the hope that it will be useful,
  24. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  25. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  26. # GNU Affero General Public License for more details.
  27. #
  28. # You should have received a copy of the GNU Affero General Public License
  29. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  30. VARIANTS='full full-vim'
  31. IN_DEFAULT_INSTALL=0
  32. SHOW_ON_ABOUT=1
  33. KEYSERVER_SKS_REPO="https://bitbucket.org/skskeyserver/sks-keyserver"
  34. KEYSERVER_SKS_COMMIT='0106ba2'
  35. KEYSERVER_WEB_REPO="https://github.com/mattrude/pgpkeyserver-lite"
  36. KEYSERVER_WEB_COMMIT='a038cb79b927c99bf7da62f20d2c6a2f20374339'
  37. KEYSERVER_PORT=11371
  38. KEYSERVER_ONION_PORT=8122
  39. KEYSERVER_DOMAIN_NAME=
  40. KEYSERVER_CODE=
  41. KEYSERVER_DUMP_URL="https://keyserver.mattrude.com/dump/current/"
  42. keyserver_variables=(ONION_ONLY
  43. MY_USERNAME
  44. DEFAULT_DOMAIN_NAME
  45. KEYSERVER_DOMAIN_NAME
  46. KEYSERVER_CODE)
  47. function logging_on_keyserver {
  48. echo -n ''
  49. }
  50. function logging_off_keyserver {
  51. echo -n ''
  52. }
  53. function reconfigure_keyserver {
  54. echo -n ''
  55. }
  56. function upgrade_keyserver_sks {
  57. CURR_KEYSERVER_SKS_COMMIT=$(get_completion_param "keyserver commit")
  58. if [[ "$CURR_KEYSERVER_SKS_COMMIT" == "$KEYSERVER_SKS_COMMIT" ]]; then
  59. return
  60. fi
  61. if grep -q "keyserver domain" $COMPLETION_FILE; then
  62. KEYSERVER_DOMAIN_NAME=$(get_completion_param "keyserver domain")
  63. fi
  64. # update to the next commit
  65. function_check set_repo_commit
  66. set_repo_commit $INSTALL_DIR/keyserver "keyserver commit" "$KEYSERVER_SKS_COMMIT" $KEYSERVER_SKS_REPO
  67. cd $INSTALL_DIR/keyserver
  68. make dep
  69. make all
  70. if [ ! "$?" = "0" ]; then
  71. echo $'Unable to build sks-keyserver'
  72. exit 836252
  73. fi
  74. make install
  75. chown -R keyserver:keyserver /var/lib/sks
  76. }
  77. function upgrade_keyserver_web {
  78. CURR_KEYSERVER_WEB_COMMIT=$(get_completion_param "keyserver web commit")
  79. if [[ "$CURR_KEYSERVER_WEB_COMMIT" == "$KEYSERVER_WEB_COMMIT" ]]; then
  80. return
  81. fi
  82. if grep -q "keyserver domain" $COMPLETION_FILE; then
  83. KEYSERVER_DOMAIN_NAME=$(get_completion_param "keyserver domain")
  84. fi
  85. # update to the next commit
  86. function_check set_repo_commit
  87. set_repo_commit /var/www/$KEYSERVER_DOMAIN_NAME/htdocs "keyserver web commit" "$KEYSERVER_WEB_COMMIT" $KEYSERVER_WEB_REPO
  88. chown -R www-data:www-data /var/www/$KEYSERVER_DOMAIN_NAME/htdocs
  89. }
  90. function upgrade_keyserver {
  91. upgrade_keyserver_sks
  92. upgrade_keyserver_web
  93. }
  94. function backup_local_keyserver {
  95. echo -n ''
  96. }
  97. function restore_local_keyserver {
  98. echo -n ''
  99. }
  100. function backup_remote_keyserver {
  101. echo -n ''
  102. }
  103. function restore_remote_keyserver {
  104. echo -n ''
  105. }
  106. function remove_keyserver {
  107. systemctl stop keyserver
  108. systemctl disable keyserver
  109. rm /etc/systemd/system/keyserver.service
  110. systemctl daemon-reload
  111. read_config_param "KEYSERVER_DOMAIN_NAME"
  112. nginx_dissite $KEYSERVER_DOMAIN_NAME
  113. remove_certs ${KEYSERVER_DOMAIN_NAME}
  114. if [ -f /etc/nginx/sites-available/$KEYSERVER_DOMAIN_NAME ]; then
  115. rm -f /etc/nginx/sites-available/$KEYSERVER_DOMAIN_NAME
  116. fi
  117. if [ -d /var/www/$KEYSERVER_DOMAIN_NAME ]; then
  118. rm -rf /var/www/$KEYSERVER_DOMAIN_NAME
  119. fi
  120. function_check remove_ddns_domain
  121. remove_ddns_domain $KEYSERVER_DOMAIN_NAME
  122. remove_config_param KEYSERVER_DOMAIN_NAME
  123. remove_config_param KEYSERVER_CODE
  124. function_check remove_onion_service
  125. remove_onion_service keyserver ${KEYSERVER_ONION_PORT}
  126. remove_completion_param "install_keyserver"
  127. sed -i '/keyserver/d' $COMPLETION_FILE
  128. if [ -f /usr/bin/keyserver-start ]; then
  129. rm /usr/bin/keyserver-start
  130. fi
  131. if [ -f /usr/bin/keyserver-stop ]; then
  132. rm /usr/bin/keyserver-stop
  133. fi
  134. cd $INSTALL_DIR/keyserver
  135. make uninstall
  136. if [ -d /var/lib/sks ]; then
  137. rm -rf /var/lib/sks
  138. fi
  139. rm -rf $INSTALL_DIR/keyserver
  140. groupdel -f keyserver
  141. userdel -r keyserver
  142. }
  143. function install_interactive_keyserver {
  144. if [ ! $ONION_ONLY ]; then
  145. ONION_ONLY='no'
  146. fi
  147. if [[ $ONION_ONLY != "no" ]]; then
  148. KEYSERVER_DOMAIN_NAME='keyserver.local'
  149. write_config_param "KEYSERVER_DOMAIN_NAME" "$KEYSERVER_DOMAIN_NAME"
  150. else
  151. function_check interactive_site_details
  152. interactive_site_details "keyserver" "KEYSERVER_DOMAIN_NAME" "KEYSERVER_CODE"
  153. fi
  154. APP_INSTALLED=1
  155. }
  156. function keyserver_import_keys {
  157. dialog --title $"Import public keys database" \
  158. --backtitle $"Freedombone Control Panel" \
  159. --defaultno \
  160. --yesno $"\nThis will download multiple gigabytes of data and so will take a long time.\n\nContinue?" 10 60
  161. sel=$?
  162. case $sel in
  163. 1) return;;
  164. 255) return;;
  165. esac
  166. if [ ! -d /var/lib/sks/dump ]; then
  167. mkdir -p /var/lib/sks/dump
  168. fi
  169. cd /var/lib/sks/dump
  170. echo $'Getting keyserver dump. This may take a few hours or longer, so be patient.'
  171. wget -crp -e robots=off --level=1 --cut-dirs=3 -nH \
  172. -A pgp,txt $KEYSERVER_DUMP_URL
  173. cd /var/lib/sks
  174. echo $'Building the keyserver database from the downloaded dump'
  175. echo '2' | /usr/local/bin/sks_build.sh
  176. }
  177. function configure_interactive_keyserver {
  178. while true
  179. do
  180. data=$(tempfile 2>/dev/null)
  181. trap "rm -f $data" 0 1 2 5 15
  182. dialog --backtitle $"Freedombone Control Panel" \
  183. --title $"SKS Keyserver" \
  184. --radiolist $"Choose an operation:" 10 70 2 \
  185. 1 $"Import public keys database" off \
  186. 2 $"Exit" on 2> $data
  187. sel=$?
  188. case $sel in
  189. 1) return;;
  190. 255) return;;
  191. esac
  192. case $(cat $data) in
  193. 1) keyserver_import_keys;;
  194. 2) break;;
  195. esac
  196. done
  197. }
  198. function install_keyserver {
  199. apt-get -qy install build-essential gcc ocaml libdb-dev wget
  200. if [ ! -d /var/www/$KEYSERVER_DOMAIN_NAME ]; then
  201. mkdir /var/www/$KEYSERVER_DOMAIN_NAME
  202. fi
  203. if [ ! -d $INSTALL_DIR ]; then
  204. mkdir -p $INSTALL_DIR
  205. fi
  206. cd $INSTALL_DIR
  207. if [ -d /repos/keyserver ]; then
  208. mkdir $INSTALL_DIR/keyserver
  209. cp -r -p /repos/keyserver/. $INSTALL_DIR/keyserver
  210. cd $INSTALL_DIR/keyserver
  211. git pull
  212. else
  213. if [ -d $INSTALL_DIR/keyserver ]; then
  214. cd $INSTALL_DIR/keyserver
  215. pull
  216. else
  217. git_clone $KEYSERVER_SKS_REPO $INSTALL_DIR/keyserver
  218. fi
  219. fi
  220. cd $INSTALL_DIR/keyserver
  221. git checkout $KEYSERVER_SKS_COMMIT -b $KEYSERVER_SKS_COMMIT
  222. set_completion_param "keyserver commit" "$KEYSERVER_SKS_COMMIT"
  223. cd /var/www/$KEYSERVER_DOMAIN_NAME
  224. if [ -d /var/www/$KEYSERVER_DOMAIN_NAME/htdocs ]; then
  225. rm -rf /var/www/$KEYSERVER_DOMAIN_NAME/htdocs
  226. fi
  227. if [ -d /repos/keyserverweb ]; then
  228. mkdir htdocs
  229. cp -r -p /repos/keyserverweb/. htdocs
  230. cd htdocs
  231. git pull
  232. else
  233. git_clone $KEYSERVER_WEB_REPO htdocs
  234. fi
  235. cd /var/www/$KEYSERVER_DOMAIN_NAME/htdocs
  236. git checkout $KEYSERVER_WEB_COMMIT -b $KEYSERVER_WEB_COMMIT
  237. set_completion_param "keyserver web commit" "$KEYSERVER_WEB_COMMIT"
  238. cd $INSTALL_DIR/keyserver
  239. if [ ! -f Makefile.local.unused ]; then
  240. echo $'Unused makefile not found'
  241. exit 72398
  242. fi
  243. cp Makefile.local.unused Makefile.local
  244. sed -i 's|LIBDB=.*|LIBDB=-ldb-5.3.1|g' Makefile.local
  245. make dep
  246. make all
  247. if [ ! "$?" = "0" ]; then
  248. echo $'Unable to build sks-keyserver'
  249. exit 8356328
  250. fi
  251. make install
  252. if [ ! -f /usr/local/bin/sks_build.sh ]; then
  253. echo $'/usr/local/bin/sks_build.sh not found'
  254. exit 238460
  255. fi
  256. USER_EMAIL_ADDRESS=$MY_USERNAME@$HOSTNAME
  257. GPG_ID=$(su -m root -c "gpg --list-keys $USER_EMAIL_ADDRESS | sed -n '2p' | sed 's/^[ \t]*//'" - $MY_USERNAME)
  258. if [ ! $GPG_ID ]; then
  259. echo $'No GPG ID for admin user'
  260. exit 846336
  261. fi
  262. if [ ${#GPG_ID} -lt 5 ]; then
  263. echo $'GPG ID not retrieved for admin user'
  264. exit 835292
  265. fi
  266. if [[ "$GPG_ID" == *"error"* ]]; then
  267. echo $'GPG ID not retrieved for admin user due to error'
  268. exit 74825
  269. fi
  270. sksconf_file=/var/lib/sks/sksconf
  271. echo 'debuglevel: 3' > $sksconf_file
  272. echo '' >> $sksconf_file
  273. echo "hostname: $KEYSERVER_DOMAIN_NAME" >> $sksconf_file
  274. echo '' >> $sksconf_file
  275. echo 'hkp_address: 127.0.0.1' >> $sksconf_file
  276. echo "hkp_port: $KEYSERVER_PORT" >> $sksconf_file
  277. echo 'recon_port: 11370' >> $sksconf_file
  278. echo '' >> $sksconf_file
  279. echo "server_contact: $GPG_ID" >> $sksconf_file
  280. echo '' >> $sksconf_file
  281. echo 'initial_stat:' >> $sksconf_file
  282. echo 'disable_mailsync:' >> $sksconf_file
  283. echo 'membership_reload_interval: 1' >> $sksconf_file
  284. echo 'stat_hour: 12' >> $sksconf_file
  285. echo '' >> $sksconf_file
  286. echo 'max_matches: 500' >> $sksconf_file
  287. KEYSERVER_ONION_HOSTNAME=$(add_onion_service keyserver 80 ${KEYSERVER_ONION_PORT})
  288. echo '#!/bin/sh' > /usr/bin/keyserver-start
  289. echo 'cd /var/lib/sks' >> /usr/bin/keyserver-start
  290. echo 'echo -n \ sks_db' >> /usr/bin/keyserver-start
  291. echo '$DAEMON db &' >> /usr/bin/keyserver-start
  292. echo 'echo -n \ sks_recon' >> /usr/bin/keyserver-start
  293. echo '$DAEMON recon &' >> /usr/bin/keyserver-start
  294. chmod +x /usr/bin/keyserver-start
  295. echo '#!/bin/sh' > /usr/bin/keyserver-stop
  296. echo 'killall sks' >> /usr/bin/keyserver-stop
  297. echo 'sleep 5' >> /usr/bin/keyserver-stop
  298. chmod +x /usr/bin/keyserver-stop
  299. echo '[Unit]' > /etc/systemd/system/keyserver.service
  300. echo 'Description=SKS Keyserver' >> /etc/systemd/system/keyserver.service
  301. echo 'After=syslog.target network.target nginx.target' >> /etc/systemd/system/keyserver.service
  302. echo '' >> /etc/systemd/system/keyserver.service
  303. echo '[Service]' >> /etc/systemd/system/keyserver.service
  304. echo 'User=keyserver' >> /etc/systemd/system/keyserver.service
  305. echo 'Group=keyserver' >> /etc/systemd/system/keyserver.service
  306. echo "WorkingDirectory=/var/lib/sks" >> /etc/systemd/system/keyserver.service
  307. echo "ExecStart=/usr/bin/keyserver-start" >> /etc/systemd/system/keyserver.service
  308. echo "ExecStop=/usr/bin/keyserver-stop" >> /etc/systemd/system/keyserver.service
  309. echo 'Restart=always' >> /etc/systemd/system/keyserver.service
  310. echo 'RestartSec=10' >> /etc/systemd/system/keyserver.service
  311. echo '' >> /etc/systemd/system/keyserver.service
  312. echo '[Install]' >> /etc/systemd/system/keyserver.service
  313. echo 'WantedBy=multi-user.target' >> /etc/systemd/system/keyserver.service
  314. chmod +x /etc/systemd/system/keyserver.service
  315. keyserver_nginx_site=/etc/nginx/sites-available/$KEYSERVER_DOMAIN_NAME
  316. if [[ $ONION_ONLY == "no" ]]; then
  317. function_check nginx_http_redirect
  318. nginx_http_redirect $KEYSERVER_DOMAIN_NAME
  319. echo 'server {' >> $keyserver_nginx_site
  320. echo ' listen 443 ssl;' >> $keyserver_nginx_site
  321. echo ' listen [::]:443 ssl;' >> $keyserver_nginx_site
  322. echo " server_name $KEYSERVER_DOMAIN_NAME;" >> $keyserver_nginx_site
  323. echo '' >> $keyserver_nginx_site
  324. echo ' # Security' >> $keyserver_nginx_site
  325. function_check nginx_ssl
  326. nginx_ssl $KEYSERVER_DOMAIN_NAME
  327. function_check nginx_disable_sniffing
  328. nginx_disable_sniffing $KEYSERVER_DOMAIN_NAME
  329. echo ' add_header Strict-Transport-Security max-age=15768000;' >> $keyserver_nginx_site
  330. echo '' >> $keyserver_nginx_site
  331. echo ' # Logs' >> $keyserver_nginx_site
  332. echo ' access_log /dev/null;' >> $keyserver_nginx_site
  333. echo ' error_log /dev/null;' >> $keyserver_nginx_site
  334. echo '' >> $keyserver_nginx_site
  335. echo ' # Root' >> $keyserver_nginx_site
  336. echo " root /var/www/$KEYSERVER_DOMAIN_NAME/htdocs;" >> $keyserver_nginx_site
  337. echo '' >> $keyserver_nginx_site
  338. echo ' rewrite ^/stats /pks/lookup?op=stats;' >> $keyserver_nginx_site
  339. echo ' rewrite ^/s/(.*) /pks/lookup?search=$1;' >> $keyserver_nginx_site
  340. echo ' rewrite ^/search/(.*) /pks/lookup?search=$1;' >> $keyserver_nginx_site
  341. echo ' rewrite ^/g/(.*) /pks/lookup?op=get&search=$1;' >> $keyserver_nginx_site
  342. echo ' rewrite ^/get/(.*) /pks/lookup?op=get&search=$1;' >> $keyserver_nginx_site
  343. echo ' rewrite ^/d/(.*) /pks/lookup?op=get&options=mr&search=$1;' >> $keyserver_nginx_site
  344. echo ' rewrite ^/download/(.*) /pks/lookup?op=get&options=mr&search=$1;' >> $keyserver_nginx_site
  345. echo '' >> $keyserver_nginx_site
  346. echo ' location /pks {' >> $keyserver_nginx_site
  347. echo " proxy_pass http://127.0.0.1:$KEYSERVER_PORT;" >> $keyserver_nginx_site
  348. echo ' proxy_pass_header Server;' >> $keyserver_nginx_site
  349. echo " add_header Via \"1.1 $KEYSERVER_DOMAIN_NAME:$KEYSERVER_PORT (nginx)\";" >> $keyserver_nginx_site
  350. echo ' proxy_ignore_client_abort on;' >> $keyserver_nginx_site
  351. echo ' client_max_body_size 8m;' >> $keyserver_nginx_site
  352. echo ' }' >> $keyserver_nginx_site
  353. echo '}' >> $keyserver_nginx_site
  354. echo '' >> $keyserver_nginx_site
  355. else
  356. echo -n '' > $keyserver_nginx_site
  357. fi
  358. echo 'server {' >> $keyserver_nginx_site
  359. echo " listen 127.0.0.1:$KEYSERVER_ONION_PORT default_server;" >> $keyserver_nginx_site
  360. echo " server_name $KEYSERVER_ONION_HOSTNAME;" >> $keyserver_nginx_site
  361. echo '' >> $keyserver_nginx_site
  362. function_check nginx_disable_sniffing
  363. nginx_disable_sniffing $KEYSERVER_DOMAIN_NAME
  364. echo '' >> $keyserver_nginx_site
  365. echo ' # Logs' >> $keyserver_nginx_site
  366. echo ' access_log /dev/null;' >> $keyserver_nginx_site
  367. echo ' error_log /dev/null;' >> $keyserver_nginx_site
  368. echo '' >> $keyserver_nginx_site
  369. echo ' # Root' >> $keyserver_nginx_site
  370. echo " root /var/www/$KEYSERVER_DOMAIN_NAME/mail;" >> $keyserver_nginx_site
  371. echo '' >> $keyserver_nginx_site
  372. echo ' rewrite ^/stats /pks/lookup?op=stats;' >> $keyserver_nginx_site
  373. echo ' rewrite ^/s/(.*) /pks/lookup?search=$1;' >> $keyserver_nginx_site
  374. echo ' rewrite ^/search/(.*) /pks/lookup?search=$1;' >> $keyserver_nginx_site
  375. echo ' rewrite ^/g/(.*) /pks/lookup?op=get&search=$1;' >> $keyserver_nginx_site
  376. echo ' rewrite ^/get/(.*) /pks/lookup?op=get&search=$1;' >> $keyserver_nginx_site
  377. echo ' rewrite ^/d/(.*) /pks/lookup?op=get&options=mr&search=$1;' >> $keyserver_nginx_site
  378. echo ' rewrite ^/download/(.*) /pks/lookup?op=get&options=mr&search=$1;' >> $keyserver_nginx_site
  379. echo '' >> $keyserver_nginx_site
  380. echo ' location /pks {' >> $keyserver_nginx_site
  381. echo " proxy_pass http://127.0.0.1:$KEYSERVER_PORT;" >> $keyserver_nginx_site
  382. echo ' proxy_pass_header Server;' >> $keyserver_nginx_site
  383. echo " add_header Via \"1.1 $KEYSERVER_DOMAIN_NAME:$KEYSERVER_PORT (nginx)\";" >> $keyserver_nginx_site
  384. echo ' proxy_ignore_client_abort on;' >> $keyserver_nginx_site
  385. echo ' client_max_body_size 8m;' >> $keyserver_nginx_site
  386. echo ' }' >> $keyserver_nginx_site
  387. echo '}' >> $keyserver_nginx_site
  388. function_check create_site_certificate
  389. if [ ! -f /etc/ssl/certs/${KEYSERVER_DOMAIN_NAME}.pem ]; then
  390. create_site_certificate $KEYSERVER_DOMAIN_NAME 'yes'
  391. fi
  392. if [ -f /etc/ssl/certs/${KEYSERVER_DOMAIN_NAME}.crt ]; then
  393. mv /etc/ssl/certs/${KEYSERVER_DOMAIN_NAME}.crt /etc/ssl/certs/${KEYSERVER_DOMAIN_NAME}.pem
  394. fi
  395. if [ -f /etc/ssl/certs/${KEYSERVER_DOMAIN_NAME}.pem ]; then
  396. chown root:root /etc/ssl/certs/${KEYSERVER_DOMAIN_NAME}.pem
  397. sed -i "s|.crt|.pem|g" /etc/nginx/sites-available/${KEYSERVER_DOMAIN_NAME}
  398. fi
  399. if [ -f /etc/ssl/private/${KEYSERVER_DOMAIN_NAME}.key ]; then
  400. chown root:root /etc/ssl/private/${KEYSERVER_DOMAIN_NAME}.key
  401. fi
  402. groupadd keyserver
  403. useradd -c "SKS Keyserver system account" -d /var/lib/sks -m -r -g keyserver keyserver
  404. chown -R keyserver:keyserver /var/lib/sks
  405. chown -R www-data:www-data /var/www/$KEYSERVER_DOMAIN_NAME/htdocs
  406. function_check nginx_ensite
  407. nginx_ensite $KEYSERVER_DOMAIN_NAME
  408. systemctl enable keyserver
  409. systemctl daemon-reload
  410. systemctl start keyserver
  411. systemctl restart nginx
  412. set_completion_param "keyserver domain" "$KEYSERVER_DOMAIN_NAME"
  413. APP_INSTALLED=1
  414. }
  415. # NOTE: deliberately no exit 0