DockerProxy_Install.sh 122 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589
  1. #!/usr/bin/env bash
  2. #===============================================================================
  3. #
  4. # FILE: DockerProxy_Install.sh
  5. #
  6. # USAGE: ./DockerProxy_Install.sh
  7. #
  8. # DESCRIPTION: 自建Docker镜像加速服务,基于官方 registry 一键部署Docker、K8s、Quay、Ghcr镜像加速\管理服务.支持部署到Render.
  9. #
  10. # ORGANIZATION: DingQz dqzboy.com 浅时光博客
  11. #===============================================================================
  12. echo
  13. cat << EOF
  14. ██████╗ ██████╗ ██████╗██╗ ██╗███████╗██████╗ ██████╗ ██████╗ ██████╗ ██╗ ██╗██╗ ██╗
  15. ██╔══██╗██╔═══██╗██╔════╝██║ ██╔╝██╔════╝██╔══██╗ ██╔══██╗██╔══██╗██╔═══██╗╚██╗██╔╝╚██╗ ██╔╝
  16. ██║ ██║██║ ██║██║ █████╔╝ █████╗ ██████╔╝ ██████╔╝██████╔╝██║ ██║ ╚███╔╝ ╚████╔╝
  17. ██║ ██║██║ ██║██║ ██╔═██╗ ██╔══╝ ██╔══██╗ ██╔═══╝ ██╔══██╗██║ ██║ ██╔██╗ ╚██╔╝
  18. ██████╔╝╚██████╔╝╚██████╗██║ ██╗███████╗██║ ██║ ██║ ██║ ██║╚██████╔╝██╔╝ ██╗ ██║
  19. ╚═════╝ ╚═════╝ ╚═════╝╚═╝ ╚═╝╚══════╝╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═╝ ╚═╝ ╚═╝
  20. 博客: dqzboy.com 浅时光博客
  21. 项目地址: https://github.com/dqzboy/Docker-Proxy
  22. EOF
  23. echo "----------------------------------------------------------------------------------------------------------"
  24. echo -e "\033[32mVPS 推荐\033[0m(\033[34mRackNerd 高性价比便宜VPS\033[0m):\033[34;4mhttps://my.racknerd.com/aff.php?aff=12151\033[0m"
  25. echo "----------------------------------------------------------------------------------------------------------"
  26. echo
  27. echo
  28. GREEN="\033[0;32m"
  29. RED="\033[31m"
  30. YELLOW="\033[33m"
  31. RESET="\033[0m"
  32. BLUE="\033[0;34m"
  33. MAGENTA="\033[0;35m"
  34. CYAN="\033[0;36m"
  35. WHITE="\033[1;37m"
  36. BLACK="\033[0;30m"
  37. PINK="\033[0;95m"
  38. LIGHT_GREEN="\033[1;32m"
  39. LIGHT_RED="\033[1;31m"
  40. LIGHT_YELLOW="\033[1;33m"
  41. LIGHT_BLUE="\033[1;34m"
  42. LIGHT_MAGENTA="\033[1;35m"
  43. LIGHT_CYAN="\033[1;36m"
  44. LIGHT_PINK="\033[1;95m"
  45. BRIGHT_CYAN="\033[96m"
  46. BOLD="\033[1m"
  47. UNDERLINE="\033[4m"
  48. BLINK="\033[5m"
  49. REVERSE="\033[7m"
  50. INFO="[${GREEN}INFO${RESET}]"
  51. ERROR="[${RED}ERROR${RESET}]"
  52. WARN="[${YELLOW}WARN${RESET}]"
  53. function INFO() {
  54. echo -e "${INFO} ${1}"
  55. }
  56. function ERROR() {
  57. echo -e "${ERROR} ${1}"
  58. }
  59. function WARN() {
  60. echo -e "${WARN} ${1}"
  61. }
  62. function PROMPT_Y_N() {
  63. echo -e "[${LIGHT_GREEN}y${RESET}/${LIGHT_BLUE}n${RESET}]: "
  64. }
  65. PROMPT_YES_NO=$(PROMPT_Y_N)
  66. function SEPARATOR() {
  67. echo -e "${INFO}${BOLD}${LIGHT_BLUE}======================== ${1} ========================${RESET}"
  68. }
  69. # 检查是否以root权限运行
  70. if [[ $EUID -ne 0 ]]; then
  71. ERROR "此脚本必须以root权限运行!"
  72. exit 1
  73. fi
  74. PROXY_DIR="/data/registry-proxy"
  75. mkdir -p ${PROXY_DIR}
  76. cd "${PROXY_DIR}"
  77. GITRAW="https://raw.githubusercontent.com/dqzboy/Docker-Proxy/main"
  78. CNGITRAW="https://gitee.com/boydqz/Docker-Proxy/raw/main"
  79. # docker registry
  80. IMAGE_NAME="dqzboy/registry"
  81. UI_IMAGE_NAME="dqzboy/docker-registry-ui"
  82. DOCKER_COMPOSE_FILE="docker-compose.yaml"
  83. # HubCMD-UI
  84. CMDUI_IMAGE_NAME="dqzboy/hubcmd-ui"
  85. CMDUI_COMPOSE_FILE="${GITRAW}/hubcmdui/${DOCKER_COMPOSE_FILE}"
  86. # Registry Domain prefix
  87. REGISTRY_SLD="ui、hub、gcr、ghcr、k8sgcr、k8s、quay、mcr、elastic、nvcr"
  88. RECORDS=("ui" "hub" "gcr" "ghcr" "k8sgcr" "k8s" "quay" "mcr" "elastic" "nvcr")
  89. attempts=0
  90. maxAttempts=3
  91. # registry services
  92. function REGISTRY_MENU() {
  93. echo -e "${YELLOW}-------------------------------------------------${RESET}"
  94. echo -e "${GREEN}1)${RESET} ${BOLD}docker hub${RESET}"
  95. echo -e "${GREEN}2)${RESET} ${BOLD}gcr${RESET}"
  96. echo -e "${GREEN}3)${RESET} ${BOLD}ghcr${RESET}"
  97. echo -e "${GREEN}4)${RESET} ${BOLD}quay${RESET}"
  98. echo -e "${GREEN}5)${RESET} ${BOLD}k8s-gcr${RESET}"
  99. echo -e "${GREEN}6)${RESET} ${BOLD}k8s${RESET}"
  100. echo -e "${GREEN}7)${RESET} ${BOLD}mcr${RESET}"
  101. echo -e "${GREEN}8)${RESET} ${BOLD}elastic${RESET}"
  102. echo -e "${GREEN}9)${RESET} ${BOLD}nvcr${RESET}"
  103. echo -e "${GREEN}10)${RESET} ${BOLD}all${RESET}"
  104. echo -e "${GREEN}0)${RESET} ${BOLD}exit${RESET}"
  105. echo -e "${YELLOW}-------------------------------------------------${RESET}"
  106. }
  107. function REGISTRY_SER_MENU() {
  108. echo -e "${YELLOW}-------------------------------------------------${RESET}"
  109. echo -e "${GREEN}1)${RESET} ${BOLD}docker hub${RESET}"
  110. echo -e "${GREEN}2)${RESET} ${BOLD}gcr${RESET}"
  111. echo -e "${GREEN}3)${RESET} ${BOLD}ghcr${RESET}"
  112. echo -e "${GREEN}4)${RESET} ${BOLD}quay${RESET}"
  113. echo -e "${GREEN}5)${RESET} ${BOLD}k8s-gcr${RESET}"
  114. echo -e "${GREEN}6)${RESET} ${BOLD}k8s${RESET}"
  115. echo -e "${GREEN}7)${RESET} ${BOLD}mcr${RESET}"
  116. echo -e "${GREEN}8)${RESET} ${BOLD}elastic${RESET}"
  117. echo -e "${GREEN}9)${RESET} ${BOLD}nvcr${RESET}"
  118. echo -e "${GREEN}0)${RESET} ${BOLD}exit${RESET}"
  119. echo -e "${YELLOW}-------------------------------------------------${RESET}"
  120. }
  121. # 定义Docker容器服务名称
  122. CONTAINER_SERVICES() {
  123. services=(
  124. "dockerhub"
  125. "gcr"
  126. "ghcr"
  127. "quay"
  128. "k8sgcr"
  129. "k8s"
  130. "mcr"
  131. "elastic"
  132. "nvcr"
  133. )
  134. }
  135. REGISTRY_FILES() {
  136. files=(
  137. "dockerhub registry-hub.yml"
  138. "gcr registry-gcr.yml"
  139. "ghcr registry-ghcr.yml"
  140. "quay registry-quay.yml"
  141. "k8sgcr registry-k8sgcr.yml"
  142. "k8s registry-k8s.yml"
  143. "mcr registry-mcr.yml"
  144. "elastic registry-elastic.yml"
  145. "nvcr registry-nvcr.yml"
  146. )
  147. }
  148. function CHECK_OS() {
  149. SEPARATOR "检查环境"
  150. OSVER=$(cat /etc/os-release | grep -o '[0-9]' | head -n 1)
  151. if [ -f /etc/os-release ]; then
  152. . /etc/os-release
  153. else
  154. echo "无法确定发行版"
  155. exit 1
  156. fi
  157. case "$ID" in
  158. "centos")
  159. repo_type="centos"
  160. ;;
  161. "debian")
  162. repo_type="debian"
  163. ;;
  164. "rhel")
  165. repo_type="rhel"
  166. ;;
  167. "ubuntu")
  168. repo_type="ubuntu"
  169. ;;
  170. "opencloudos")
  171. repo_type="centos"
  172. ;;
  173. "rocky")
  174. repo_type="centos"
  175. ;;
  176. *)
  177. WARN "此脚本目前不支持您的系统: $ID"
  178. exit 1
  179. ;;
  180. esac
  181. INFO "System release:: $NAME"
  182. INFO "System version: $VERSION"
  183. INFO "System ID: $ID"
  184. INFO "System ID Like: $ID_LIKE"
  185. }
  186. function CHECK_PACKAGE_MANAGER() {
  187. if command -v dnf &> /dev/null; then
  188. package_manager="dnf"
  189. elif command -v yum &> /dev/null; then
  190. package_manager="yum"
  191. elif command -v apt-get &> /dev/null; then
  192. package_manager="apt-get"
  193. elif command -v apt &> /dev/null; then
  194. package_manager="apt"
  195. else
  196. ERROR "不受支持的软件包管理器."
  197. exit 1
  198. fi
  199. }
  200. function CHECK_PKG_MANAGER() {
  201. if command -v rpm &> /dev/null; then
  202. pkg_manager="rpm"
  203. elif command -v dpkg &> /dev/null; then
  204. pkg_manager="dpkg"
  205. elif command -v apt &> /dev/null; then
  206. pkg_manager="apt"
  207. else
  208. ERROR "无法确定包管理系统."
  209. exit 1
  210. fi
  211. }
  212. function CHECK_COMPOSE_CMD() {
  213. if command -v docker &> /dev/null && docker compose version &> /dev/null; then
  214. DOCKER_COMPOSE_CMD="docker compose"
  215. elif command -v docker-compose &> /dev/null; then
  216. DOCKER_COMPOSE_CMD="docker-compose"
  217. else
  218. WARN "未检查到Docker Compose客户端工具,请通过脚本安装部署!"
  219. fi
  220. }
  221. function CHECKMEM() {
  222. memory_usage=$(free | awk '/^Mem:/ {printf "%.2f", $3/$2 * 100}')
  223. memory_usage=${memory_usage%.*}
  224. if [[ $memory_usage -gt 90 ]]; then
  225. read -e -p "$(WARN "内存占用率${LIGHT_RED}高于 70%($memory_usage%)${RESET} 是否继续安装? ${PROMPT_YES_NO}")" continu
  226. if [ "$continu" == "n" ] || [ "$continu" == "N" ]; then
  227. exit 1
  228. fi
  229. else
  230. INFO "内存资源充足.请继续 ${LIGHT_GREEN}($memory_usage%)${RESET}"
  231. fi
  232. }
  233. function CHECKFIRE() {
  234. systemctl stop firewalld &> /dev/null
  235. systemctl disable firewalld &> /dev/null
  236. systemctl stop iptables &> /dev/null
  237. systemctl disable iptables &> /dev/null
  238. ufw disable &> /dev/null
  239. INFO "防火墙已被禁用."
  240. if [[ "$repo_type" == "centos" || "$repo_type" == "rhel" ]]; then
  241. if sestatus | grep "SELinux status" | grep -q "enabled"; then
  242. WARN "SELinux 已启用。禁用 SELinux..."
  243. setenforce 0
  244. sed -i 's/SELINUX=enforcing/SELINUX=disabled/g' /etc/selinux/config
  245. INFO "SELinux 已被禁用."
  246. else
  247. INFO "SELinux 已被禁用."
  248. fi
  249. fi
  250. }
  251. function CHECKBBR() {
  252. kernel_version=$(uname -r | awk -F "-" '{print $1}')
  253. read -e -p "$(WARN "是否开启${BRIGHT_CYAN}BBR${RESET},优化网络带宽提高网络性能? ${PROMPT_YES_NO}")" choice_bbr
  254. case $choice_bbr in
  255. y | Y)
  256. version_compare=$(echo "${kernel_version} 4.9" | awk '{if ($1 >= $2) print "yes"; else print "no"}')
  257. if [ "$version_compare" != "yes" ]; then
  258. WARN "你的内核版本小于4.9,无法启动BBR,需要你手动升级内核"
  259. exit 0
  260. fi
  261. sysctl net.ipv4.tcp_available_congestion_control | grep -q "bbr"
  262. if [ $? -eq 0 ]; then
  263. INFO "你的服务器已经启动 ${BRIGHT_CYAN}BBR${RESET}"
  264. else
  265. INFO "开启BBR中..."
  266. modprobe tcp_bbr
  267. if [ $? -eq 0 ]; then
  268. INFO "${BRIGHT_CYAN}BBR${RESET} 模块${LIGHT_GREEN}添加成功${RESET}"
  269. else
  270. ERROR "${BRIGHT_CYAN}BBR${RESET} 模块${LIGHT_RED}添加失败${RESET},请执行 ${LIGHT_CYAN}sysctl -p${RESET} 检查."
  271. exit 1
  272. fi
  273. if [ ! -d /etc/modules-load.d/ ]; then
  274. mkdir -p /etc/modules-load.d/
  275. fi
  276. if [ ! -f /etc/modules-load.d/tcp_bbr.conf ]; then
  277. touch /etc/modules-load.d/tcp_bbr.conf
  278. fi
  279. if ! grep -q "tcp_bbr" /etc/modules-load.d/tcp_bbr.conf ; then
  280. echo 'tcp_bbr' >> /etc/modules-load.d/tcp_bbr.conf
  281. fi
  282. for setting in "net.core.default_qdisc=fq" "net.ipv4.tcp_congestion_control=bbr"; do
  283. if ! grep -q "$setting" /etc/sysctl.conf; then
  284. echo "$setting" >> /etc/sysctl.conf
  285. fi
  286. done
  287. sysctl -p &> /dev/null
  288. if [ $? -ne 0 ]; then
  289. ERROR "应用sysctl设置过程中发生了一个错误,请执行 ${LIGHT_CYAN}sysctl -p${RESET} 检查."
  290. exit 2
  291. fi
  292. lsmod | grep tcp_bbr &> /dev/null
  293. if [ $? -eq 0 ]; then
  294. INFO "${BRIGHT_CYAN}BBR${RESET} 已经${LIGHT_GREEN}成功开启${RESET}"
  295. else
  296. ERROR "${BRIGHT_CYAN}BBR${RESET} 开启${LIGHT_RED}失败${RESET},请执行 ${LIGHT_CYAN}sysctl -p${RESET} 检查."
  297. exit 3
  298. fi
  299. WARN "如果 ${BRIGHT_CYAN}BBR${RESET} 开启后${LIGHT_YELLOW}未生效${RESET},请执行 ${LIGHT_BLUE}reboot${RESET} 重启服务器使其BBR模块生效"
  300. fi
  301. ;;
  302. n | N)
  303. INFO "不开启BBR"
  304. ;;
  305. *)
  306. WARN "输入了无效的选择。请重新输入${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}"
  307. CHECKBBR
  308. ;;
  309. esac
  310. }
  311. function INSTALL_PACKAGE(){
  312. SEPARATOR "安装依赖"
  313. INFO "检查依赖安装情况,请稍等 ..."
  314. TIMEOUT=300
  315. PACKAGES_APT=(
  316. lsof jq wget apache2-utils tar
  317. )
  318. PACKAGES_YUM=(
  319. epel-release lsof jq wget yum-utils httpd-tools tar
  320. )
  321. if [ "$package_manager" = "dnf" ] || [ "$package_manager" = "yum" ]; then
  322. for package in "${PACKAGES_YUM[@]}"; do
  323. if $pkg_manager -q "$package" &>/dev/null; then
  324. INFO "${LIGHT_GREEN}已经安装${RESET} $package ..."
  325. else
  326. INFO "${LIGHT_CYAN}正在安装${RESET} $package ..."
  327. start_time=$(date +%s)
  328. $package_manager -y install "$package" --skip-broken > /dev/null 2>&1 &
  329. install_pid=$!
  330. while [[ $(($(date +%s) - $start_time)) -lt $TIMEOUT ]] && kill -0 $install_pid &>/dev/null; do
  331. sleep 1
  332. done
  333. if kill -0 $install_pid &>/dev/null; then
  334. read -e -p "$(WARN "$package 的安装时间超过 ${LIGHT_YELLOW}$TIMEOUT 秒${RESET}。是否继续? ${PROMPT_YES_NO}")" continue_install
  335. if [ "$continue_install" != "y" ]; then
  336. ERROR "$package 的安装超时。退出脚本。"
  337. exit 1
  338. else
  339. continue
  340. fi
  341. fi
  342. wait $install_pid
  343. if [ $? -ne 0 ]; then
  344. ERROR "$package 安装失败。请检查系统安装源,然后再次运行此脚本!请尝试手动执行安装: ${LIGHT_BLUE}$package_manager -y install $package${RESET}"
  345. exit 1
  346. fi
  347. fi
  348. done
  349. elif [ "$package_manager" = "apt-get" ] || [ "$package_manager" = "apt" ];then
  350. dpkg --configure -a &>/dev/null
  351. $package_manager update &>/dev/null
  352. for package in "${PACKAGES_APT[@]}"; do
  353. if $pkg_manager -s "$package" &>/dev/null; then
  354. INFO "已经安装 $package ..."
  355. else
  356. INFO "正在安装 $package ..."
  357. $package_manager install -y $package > /dev/null 2>&1
  358. if [ $? -ne 0 ]; then
  359. ERROR "安装 $package 失败,请检查系统安装源之后再次运行此脚本!请尝试手动执行安装: ${LIGHT_BLUE}$package_manager -y install $package${RESET}"
  360. exit 1
  361. fi
  362. fi
  363. done
  364. else
  365. ERROR "无法确定包管理系统,脚本无法继续执行,请检查!"
  366. exit 1
  367. fi
  368. }
  369. function INSTALL_CADDY() {
  370. SEPARATOR "安装Caddy"
  371. start_caddy() {
  372. systemctl enable caddy.service &>/dev/null
  373. systemctl restart caddy.service
  374. status=$(systemctl is-active caddy)
  375. if [ "$status" = "active" ]; then
  376. INFO "Caddy 服务运行正常,请继续..."
  377. else
  378. ERROR "Caddy 服务未运行,请查看日志报错,定位问题后再次执行脚本!"
  379. ERROR "-----------服务启动失败,请查看错误日志 ↓↓↓-----------"
  380. journalctl -u caddy.service --no-pager
  381. ERROR "-----------服务启动失败,请查看错误日志 ↑↑↑-----------"
  382. exit 1
  383. fi
  384. }
  385. check_caddy() {
  386. if pgrep "caddy" > /dev/null; then
  387. INFO "Caddy 已在运行."
  388. else
  389. WARN "Caddy 未运行。尝试启动 Caddy..."
  390. start_attempts=3
  391. for ((i=1; i<=$start_attempts; i++)); do
  392. start_caddy
  393. if pgrep "caddy" > /dev/null; then
  394. INFO "Caddy 已成功启动."
  395. break
  396. else
  397. if [ $i -eq $start_attempts ]; then
  398. ERROR "Caddy 在尝试 $start_attempts 后无法启动。请检查配置"
  399. exit 1
  400. else
  401. WARN "在 $i 时间内启动 Caddy 失败。重试..."
  402. fi
  403. fi
  404. done
  405. fi
  406. }
  407. if [ "$package_manager" = "dnf" ]; then
  408. if which caddy &>/dev/null; then
  409. INFO "Caddy 已经安装."
  410. else
  411. INFO "正在安装Caddy程序,请稍候..."
  412. $package_manager -y install 'dnf-command(copr)' &>/dev/null
  413. $package_manager -y copr enable @caddy/caddy &>/dev/null
  414. while [ $attempts -lt $maxAttempts ]; do
  415. $package_manager -y install caddy &>/dev/null
  416. if [ $? -ne 0 ]; then
  417. ((attempts++))
  418. WARN "正在尝试安装Caddy >>> (Attempt: $attempts)"
  419. if [ $attempts -eq $maxAttempts ]; then
  420. ERROR "Caddy installation failed. Please try installing manually."
  421. echo "命令: $package_manager -y install 'dnf-command(copr)' && $package_manager -y copr enable @caddy/caddy && $package_manager -y install caddy"
  422. exit 1
  423. fi
  424. else
  425. INFO "已安装 Caddy."
  426. break
  427. fi
  428. done
  429. fi
  430. check_caddy
  431. elif [ "$package_manager" = "yum" ]; then
  432. if which caddy &>/dev/null; then
  433. INFO "Caddy 已经安装."
  434. else
  435. INFO "正在安装Caddy程序,请稍候..."
  436. $package_manager -y install yum-plugin-copr &>/dev/null
  437. $package_manager -y copr enable @caddy/caddy &>/dev/null
  438. while [ $attempts -lt $maxAttempts ]; do
  439. $package_manager -y install caddy &>/dev/null
  440. if [ $? -ne 0 ]; then
  441. ((attempts++))
  442. WARN "正在尝试安装Caddy >>> (Attempt: $attempts)"
  443. if [ $attempts -eq $maxAttempts ]; then
  444. ERROR "Caddy installation failed. Please try installing manually."
  445. echo "命令: $package_manager -y install 'dnf-command(copr)' && $package_manager -y copr enable @caddy/caddy && $package_manager -y install caddy"
  446. exit 1
  447. fi
  448. else
  449. INFO "已安装 Caddy."
  450. break
  451. fi
  452. done
  453. fi
  454. check_caddy
  455. elif [ "$package_manager" = "apt" ] || [ "$package_manager" = "apt-get" ];then
  456. dpkg --configure -a &>/dev/null
  457. $package_manager update &>/dev/null
  458. if $pkg_manager -s "caddy" &>/dev/null; then
  459. INFO "Caddy 已安装,跳过..."
  460. else
  461. INFO "安装 Caddy 请稍等 ..."
  462. $package_manager install -y debian-keyring debian-archive-keyring apt-transport-https &>/dev/null
  463. curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | sudo gpg --dearmor -o /usr/share/keyrings/caddy-stable-archive-keyring.gpg &>/dev/null
  464. curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/debian.deb.txt' | sudo tee /etc/apt/sources.list.d/caddy-stable.list &>/dev/null
  465. $package_manager update &>/dev/null
  466. $package_manager install -y caddy &>/dev/null
  467. if [ $? -ne 0 ]; then
  468. ERROR "安装 Caddy 失败,请检查系统安装源之后再次运行此脚本!请尝试手动执行安装:$package_manager -y install caddy"
  469. exit 1
  470. fi
  471. fi
  472. check_caddy
  473. else
  474. WARN "无法确定包管理系统."
  475. exit 1
  476. fi
  477. }
  478. function CONFIG_CADDY() {
  479. SEPARATOR "配置Caddy"
  480. while true; do
  481. INFO "${LIGHT_GREEN}>>> 域名解析主机记录(即域名前缀):${RESET} ${LIGHT_CYAN}${REGISTRY_SLD}${RESET}"
  482. WARN "${LIGHT_GREEN}>>> 只需选择你部署的服务进行解析即可${RESET},${LIGHT_YELLOW}无需将上面提示中所有的主机记录进行解析${RESET}"
  483. read -e -p "$(WARN "是否配置Caddy,实现自动HTTPS? 执行前需在DNS服务商对部署服务解析主机记录 ${PROMPT_YES_NO}")" caddy_conf
  484. case "$caddy_conf" in
  485. y|Y )
  486. read -e -p "$(INFO "请输入你的域名${LIGHT_BLUE}[例: baidu.com]${RESET} ${LIGHT_RED}不可为空${RESET}: ")" caddy_domain
  487. read -e -p "$(INFO "请输入要配置的${LIGHT_MAGENTA}主机记录${RESET},用逗号分隔${LIGHT_BLUE}[例: ui,hub]${RESET}: ")" selected_records
  488. # 验证输入的主机记录
  489. local valid_records=("${RECORDS[@]}")
  490. IFS=',' read -r -a records_array <<< "$selected_records"
  491. local invalid_records=()
  492. for record in "${records_array[@]}"; do
  493. if ! [[ " ${valid_records[@]} " =~ " ${record} " ]]; then
  494. invalid_records+=("$record")
  495. fi
  496. done
  497. if [[ ${#invalid_records[@]} -gt 0 ]]; then
  498. ERROR "无效的主机记录: ${LIGHT_RED}${invalid_records[@]}${RESET}"
  499. INFO "请输入有效的主机记录: ${LIGHT_GREEN}${REGISTRY_SLD}${RESET}"
  500. continue
  501. fi
  502. declare -A record_templates
  503. record_templates[ui]="ui.$caddy_domain {
  504. reverse_proxy localhost:50000 {
  505. header_up Host {host}
  506. header_up Origin {scheme}://{host}
  507. header_up X-Forwarded-For {remote_addr}
  508. header_up X-Forwarded-Proto {scheme}
  509. header_up X-Forwarded-Ssl on
  510. header_up X-Forwarded-Port {server_port}
  511. header_up X-Forwarded-Host {host}
  512. }
  513. }"
  514. record_templates[hub]="hub.$caddy_domain {
  515. reverse_proxy localhost:51000 {
  516. header_up Host {host}
  517. header_up X-Real-IP {remote_addr}
  518. header_up X-Forwarded-For {remote_addr}
  519. header_up X-Nginx-Proxy true
  520. }
  521. }"
  522. record_templates[ghcr]="ghcr.$caddy_domain {
  523. reverse_proxy localhost:52000 {
  524. header_up Host {host}
  525. header_up X-Real-IP {remote_addr}
  526. header_up X-Forwarded-For {remote_addr}
  527. header_up X-Nginx-Proxy true
  528. }
  529. }"
  530. record_templates[gcr]="gcr.$caddy_domain {
  531. reverse_proxy localhost:53000 {
  532. header_up Host {host}
  533. header_up X-Real-IP {remote_addr}
  534. header_up X-Forwarded-For {remote_addr}
  535. header_up X-Nginx-Proxy true
  536. }
  537. }"
  538. record_templates[k8sgcr]="k8sgcr.$caddy_domain {
  539. reverse_proxy localhost:54000 {
  540. header_up Host {host}
  541. header_up X-Real-IP {remote_addr}
  542. header_up X-Forwarded-For {remote_addr}
  543. header_up X-Nginx-Proxy true
  544. }
  545. }"
  546. record_templates[k8s]="k8s.$caddy_domain {
  547. reverse_proxy localhost:55000 {
  548. header_up Host {host}
  549. header_up X-Real-IP {remote_addr}
  550. header_up X-Forwarded-For {remote_addr}
  551. header_up X-Nginx-Proxy true
  552. }
  553. }"
  554. record_templates[quay]="quay.$caddy_domain {
  555. reverse_proxy localhost:56000 {
  556. header_up Host {host}
  557. header_up X-Real-IP {remote_addr}
  558. header_up X-Forwarded-For {remote_addr}
  559. header_up X-Nginx-Proxy true
  560. }
  561. }"
  562. record_templates[mcr]="mcr.$caddy_domain {
  563. reverse_proxy localhost:57000 {
  564. header_up Host {host}
  565. header_up X-Real-IP {remote_addr}
  566. header_up X-Forwarded-For {remote_addr}
  567. header_up X-Nginx-Proxy true
  568. }
  569. }"
  570. record_templates[elastic]="elastic.$caddy_domain {
  571. reverse_proxy localhost:58000 {
  572. header_up Host {host}
  573. header_up X-Real-IP {remote_addr}
  574. header_up X-Forwarded-For {remote_addr}
  575. header_up X-Nginx-Proxy true
  576. }
  577. }"
  578. record_templates[nvcr]="nvcr.$caddy_domain {
  579. reverse_proxy localhost:59000 {
  580. header_up Host {host}
  581. header_up X-Real-IP {remote_addr}
  582. header_up X-Forwarded-For {remote_addr}
  583. header_up X-Nginx-Proxy true
  584. }
  585. }"
  586. > /etc/caddy/Caddyfile
  587. for record in "${records_array[@]}"; do
  588. if [[ -n "${record_templates[$record]}" ]]; then
  589. echo "${record_templates[$record]}" >> /etc/caddy/Caddyfile
  590. fi
  591. done
  592. start_attempts=3
  593. for ((i=1; i<=$start_attempts; i++)); do
  594. start_caddy
  595. if pgrep "caddy" > /dev/null; then
  596. INFO "重新载入配置成功. Caddy服务启动完成"
  597. break
  598. else
  599. if [ $i -eq $start_attempts ]; then
  600. ERROR "Caddy 在尝试 $start_attempts 后无法启动。请检查配置"
  601. exit 1
  602. else
  603. WARN "第 $i 次启动 Caddy 失败。重试..."
  604. fi
  605. fi
  606. done
  607. break;;
  608. n|N )
  609. WARN "退出配置 Caddy 操作。"
  610. break;;
  611. * )
  612. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}";;
  613. esac
  614. done
  615. }
  616. function INSTALL_NGINX() {
  617. SEPARATOR "安装Nginx"
  618. start_nginx() {
  619. systemctl enable nginx &>/dev/null
  620. systemctl restart nginx
  621. status=$(systemctl is-active nginx)
  622. if [ "$status" = "active" ]; then
  623. INFO "Nginx 服务运行正常,请继续..."
  624. else
  625. ERROR "Nginx 服务未运行,请查看日志报错,定位问题后再次执行脚本!"
  626. ERROR "-----------服务启动失败,请查看错误日志 ↓↓↓-----------"
  627. journalctl -u nginx.service --no-pager
  628. ERROR "-----------服务启动失败,请查看错误日志 ↑↑↑-----------"
  629. exit 1
  630. fi
  631. }
  632. check_nginx() {
  633. if pgrep "nginx" > /dev/null; then
  634. INFO "Nginx 已在运行."
  635. else
  636. WARN "Nginx 未运行。尝试启动 Nginx..."
  637. start_attempts=3
  638. for ((i=1; i<=$start_attempts; i++)); do
  639. start_nginx
  640. if pgrep "nginx" > /dev/null; then
  641. INFO "Nginx 已成功启动."
  642. break
  643. else
  644. if [ $i -eq $start_attempts ]; then
  645. ERROR "Nginx 在尝试 $start_attempts 次后无法启动。请检查配置"
  646. exit 1
  647. else
  648. WARN "第 $i 次启动 Nginx 失败。重试..."
  649. fi
  650. fi
  651. done
  652. fi
  653. }
  654. if [ "$package_manager" = "dnf" ] || [ "$package_manager" = "yum" ]; then
  655. if which nginx &>/dev/null; then
  656. INFO "Nginx 已经安装."
  657. else
  658. INFO "正在安装Nginx程序,请稍候..."
  659. NGINX="nginx-1.24.0-1.el${OSVER}.ngx.x86_64.rpm"
  660. rm -f ${NGINX}
  661. wget http://nginx.org/packages/centos/${OSVER}/x86_64/RPMS/${NGINX} &>/dev/null
  662. while [ $attempts -lt $maxAttempts ]; do
  663. $package_manager -y install ${NGINX} &>/dev/null
  664. if [ $? -ne 0 ]; then
  665. ((attempts++))
  666. WARN "正在尝试安装Nginx >>> (Attempt: $attempts)"
  667. if [ $attempts -eq $maxAttempts ]; then
  668. ERROR "Nginx installation failed. Please try installing manually."
  669. rm -f ${NGINX}
  670. echo "命令: wget http://nginx.org/packages/centos/${OSVER}/x86_64/RPMS/${NGINX} && $package_manager -y install ${NGINX}"
  671. exit 1
  672. fi
  673. else
  674. INFO "已安装 Nginx."
  675. rm -f ${NGINX}
  676. break
  677. fi
  678. done
  679. fi
  680. check_nginx
  681. elif [ "$package_manager" = "apt-get" ] || [ "$package_manager" = "apt" ];then
  682. dpkg --configure -a &>/dev/null
  683. $package_manager update &>/dev/null
  684. if $pkg_manager -s "nginx" &>/dev/null; then
  685. INFO "nginx 已安装,跳过..."
  686. else
  687. INFO "安装 nginx 请稍等 ..."
  688. $package_manager install -y nginx > /dev/null 2>&1
  689. if [ $? -ne 0 ]; then
  690. ERROR "安装 nginx 失败,请检查系统安装源之后再次运行此脚本!请尝试手动执行安装:$package_manager -y install nginx"
  691. exit 1
  692. fi
  693. fi
  694. check_nginx
  695. else
  696. WARN "无法确定包管理系统."
  697. exit 1
  698. fi
  699. }
  700. function CONFIG_NGINX() {
  701. SEPARATOR "配置Nginx"
  702. while true; do
  703. WARN "自行安装的 Nginx ${LIGHT_RED}请勿执行此操作${RESET},${LIGHT_BLUE}以防覆盖原有配置${RESET}"
  704. INFO "${LIGHT_GREEN}>>> 域名解析主机记录(即域名前缀):${RESET} ${LIGHT_CYAN}${REGISTRY_SLD}${RESET}"
  705. WARN "${LIGHT_GREEN}>>> 只需选择你部署的服务进行解析即可${RESET},${LIGHT_YELLOW}无需将上面提示中所有的主机记录进行解析${RESET}"
  706. read -e -p "$(WARN "是否配置 Nginx?配置完成后需在DNS服务商解析主机记录 ${PROMPT_YES_NO}")" nginx_conf
  707. case "$nginx_conf" in
  708. y|Y )
  709. read -e -p "$(INFO "请输入你的域名${LIGHT_BLUE}[例: baidu.com]${RESET} ${LIGHT_RED}不可为空${RESET}: ")" nginx_domain
  710. read -e -p "$(INFO "请输入要配置的${LIGHT_MAGENTA}主机记录${RESET},用逗号分隔${LIGHT_BLUE}[例: ui,hub]${RESET}: ")" selected_records
  711. # 验证输入的主机记录
  712. local valid_records=("${RECORDS[@]}")
  713. IFS=',' read -r -a records_array <<< "$selected_records"
  714. local invalid_records=()
  715. for record in "${records_array[@]}"; do
  716. if ! [[ " ${valid_records[@]} " =~ " ${record} " ]]; then
  717. invalid_records+=("$record")
  718. fi
  719. done
  720. if [[ ${#invalid_records[@]} -gt 0 ]]; then
  721. ERROR "无效的主机记录: ${LIGHT_RED}${invalid_records[@]}${RESET}"
  722. INFO "请输入有效的主机记录: ${LIGHT_GREEN}${REGISTRY_SLD}${RESET}"
  723. continue
  724. fi
  725. declare -A record_templates
  726. record_templates[ui]="server {
  727. listen 80;
  728. #listen 443 ssl;
  729. server_name ui.$nginx_domain;
  730. #ssl_certificate /path/to/your_domain_name.crt;
  731. #ssl_certificate_key /path/to/your_domain_name.key;
  732. #ssl_session_timeout 1d;
  733. #ssl_session_cache shared:SSL:50m;
  734. #ssl_session_tickets off;
  735. #ssl_protocols TLSv1.2 TLSv1.3;
  736. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  737. #ssl_prefer_server_ciphers on;
  738. #ssl_buffer_size 8k;
  739. proxy_connect_timeout 600;
  740. proxy_send_timeout 600;
  741. proxy_read_timeout 600;
  742. send_timeout 600;
  743. location / {
  744. proxy_pass http://localhost:50000;
  745. proxy_set_header Host \$host;
  746. proxy_set_header Origin \$scheme://\$host;
  747. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  748. proxy_set_header X-Forwarded-Proto \$scheme;
  749. proxy_set_header X-Forwarded-Ssl on;
  750. proxy_set_header X-Forwarded-Port \$server_port;
  751. proxy_set_header X-Forwarded-Host \$host;
  752. }
  753. }"
  754. record_templates[hub]="server {
  755. listen 80;
  756. #listen 443 ssl;
  757. server_name hub.$nginx_domain;
  758. #ssl_certificate /path/to/your_domain_name.crt;
  759. #ssl_certificate_key /path/to/your_domain_name.key;
  760. #ssl_session_timeout 1d;
  761. #ssl_session_cache shared:SSL:50m;
  762. #ssl_session_tickets off;
  763. #ssl_protocols TLSv1.2 TLSv1.3;
  764. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  765. #ssl_prefer_server_ciphers on;
  766. #ssl_buffer_size 8k;
  767. proxy_connect_timeout 600;
  768. proxy_send_timeout 600;
  769. proxy_read_timeout 600;
  770. send_timeout 600;
  771. location / {
  772. proxy_pass http://localhost:51000;
  773. proxy_set_header Host \$host;
  774. proxy_set_header X-Real-IP \$remote_addr;
  775. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  776. proxy_set_header X-Nginx-Proxy true;
  777. proxy_buffering off;
  778. proxy_redirect off;
  779. }
  780. }"
  781. record_templates[ghcr]="server {
  782. listen 80;
  783. #listen 443 ssl;
  784. server_name ghcr.$nginx_domain;
  785. #ssl_certificate /path/to/your_domain_name.crt;
  786. #ssl_certificate_key /path/to/your_domain_name.key;
  787. #ssl_session_timeout 1d;
  788. #ssl_session_cache shared:SSL:50m;
  789. #ssl_session_tickets off;
  790. #ssl_protocols TLSv1.2 TLSv1.3;
  791. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  792. #ssl_prefer_server_ciphers on;
  793. #ssl_buffer_size 8k;
  794. proxy_connect_timeout 600;
  795. proxy_send_timeout 600;
  796. proxy_read_timeout 600;
  797. send_timeout 600;
  798. location / {
  799. proxy_pass http://localhost:52000;
  800. proxy_set_header Host \$host;
  801. proxy_set_header X-Real-IP \$remote_addr;
  802. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  803. proxy_set_header X-Nginx-Proxy true;
  804. proxy_buffering off;
  805. proxy_redirect off;
  806. }
  807. }"
  808. record_templates[gcr]="server {
  809. listen 80;
  810. #listen 443 ssl;
  811. server_name gcr.$nginx_domain;
  812. #ssl_certificate /path/to/your_domain_name.crt;
  813. #ssl_certificate_key /path/to/your_domain_name.key;
  814. #ssl_session_timeout 1d;
  815. #ssl_session_cache shared:SSL:50m;
  816. #ssl_session_tickets off;
  817. #ssl_protocols TLSv1.2 TLSv1.3;
  818. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  819. #ssl_prefer_server_ciphers on;
  820. #ssl_buffer_size 8k;
  821. proxy_connect_timeout 600;
  822. proxy_send_timeout 600;
  823. proxy_read_timeout 600;
  824. send_timeout 600;
  825. location / {
  826. proxy_pass http://localhost:53000;
  827. proxy_set_header Host \$host;
  828. proxy_set_header X-Real-IP \$remote_addr;
  829. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  830. proxy_set_header X-Nginx-Proxy true;
  831. proxy_buffering off;
  832. proxy_redirect off;
  833. }
  834. }"
  835. record_templates[k8sgcr]="server {
  836. listen 80;
  837. #listen 443 ssl;
  838. server_name k8sgcr.$nginx_domain;
  839. #ssl_certificate /path/to/your_domain_name.crt;
  840. #ssl_certificate_key /path/to/your_domain_name.key;
  841. #ssl_session_timeout 1d;
  842. #ssl_session_cache shared:SSL:50m;
  843. #ssl_session_tickets off;
  844. #ssl_protocols TLSv1.2 TLSv1.3;
  845. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  846. #ssl_prefer_server_ciphers on;
  847. #ssl_buffer_size 8k;
  848. proxy_connect_timeout 600;
  849. proxy_send_timeout 600;
  850. proxy_read_timeout 600;
  851. send_timeout 600;
  852. location / {
  853. proxy_pass http://localhost:54000;
  854. proxy_set_header Host \$host;
  855. proxy_set_header X-Real-IP \$remote_addr;
  856. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  857. proxy_set_header X-Nginx-Proxy true;
  858. proxy_buffering off;
  859. proxy_redirect off;
  860. }
  861. }"
  862. record_templates[k8s]="server {
  863. listen 80;
  864. #listen 443 ssl;
  865. server_name k8s.$nginx_domain;
  866. #ssl_certificate /path/to/your_domain_name.crt;
  867. #ssl_certificate_key /path/to/your_domain_name.key;
  868. #ssl_session_timeout 1d;
  869. #ssl_session_cache shared:SSL:50m;
  870. #ssl_session_tickets off;
  871. #ssl_protocols TLSv1.2 TLSv1.3;
  872. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  873. #ssl_prefer_server_ciphers on;
  874. #ssl_buffer_size 8k;
  875. proxy_connect_timeout 600;
  876. proxy_send_timeout 600;
  877. proxy_read_timeout 600;
  878. send_timeout 600;
  879. location / {
  880. proxy_pass http://localhost:55000;
  881. proxy_set_header Host \$host;
  882. proxy_set_header X-Real-IP \$remote_addr;
  883. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  884. proxy_set_header X-Nginx-Proxy true;
  885. proxy_buffering off;
  886. proxy_redirect off;
  887. }
  888. }"
  889. record_templates[quay]="server {
  890. listen 80;
  891. #listen 443 ssl;
  892. server_name quay.$nginx_domain;
  893. #ssl_certificate /path/to/your_domain_name.crt;
  894. #ssl_certificate_key /path/to/your_domain_name.key;
  895. #ssl_session_timeout 1d;
  896. #ssl_session_cache shared:SSL:50m;
  897. #ssl_session_tickets off;
  898. #ssl_protocols TLSv1.2 TLSv1.3;
  899. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  900. #ssl_prefer_server_ciphers on;
  901. #ssl_buffer_size 8k;
  902. proxy_connect_timeout 600;
  903. proxy_send_timeout 600;
  904. proxy_read_timeout 600;
  905. send_timeout 600;
  906. location / {
  907. proxy_pass http://localhost:56000;
  908. proxy_set_header Host \$host;
  909. proxy_set_header X-Real-IP \$remote_addr;
  910. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  911. proxy_set_header X-Nginx-Proxy true;
  912. proxy_buffering off;
  913. proxy_redirect off;
  914. }
  915. }"
  916. record_templates[mcr]="server {
  917. listen 80;
  918. #listen 443 ssl;
  919. server_name mcr.$nginx_domain;
  920. #ssl_certificate /path/to/your_domain_name.crt;
  921. #ssl_certificate_key /path/to/your_domain_name.key;
  922. #ssl_session_timeout 1d;
  923. #ssl_session_cache shared:SSL:50m;
  924. #ssl_session_tickets off;
  925. #ssl_protocols TLSv1.2 TLSv1.3;
  926. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  927. #ssl_prefer_server_ciphers on;
  928. #ssl_buffer_size 8k;
  929. proxy_connect_timeout 600;
  930. proxy_send_timeout 600;
  931. proxy_read_timeout 600;
  932. send_timeout 600;
  933. location / {
  934. proxy_pass http://localhost:57000;
  935. proxy_set_header Host \$host;
  936. proxy_set_header X-Real-IP \$remote_addr;
  937. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  938. proxy_set_header X-Nginx-Proxy true;
  939. proxy_buffering off;
  940. proxy_redirect off;
  941. }
  942. }"
  943. record_templates[elastic]="server {
  944. listen 80;
  945. #listen 443 ssl;
  946. server_name elastic.$nginx_domain;
  947. #ssl_certificate /path/to/your_domain_name.crt;
  948. #ssl_certificate_key /path/to/your_domain_name.key;
  949. #ssl_session_timeout 1d;
  950. #ssl_session_cache shared:SSL:50m;
  951. #ssl_session_tickets off;
  952. #ssl_protocols TLSv1.2 TLSv1.3;
  953. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  954. #ssl_prefer_server_ciphers on;
  955. #ssl_buffer_size 8k;
  956. proxy_connect_timeout 600;
  957. proxy_send_timeout 600;
  958. proxy_read_timeout 600;
  959. send_timeout 600;
  960. location / {
  961. proxy_pass http://localhost:58000;
  962. proxy_set_header Host \$host;
  963. proxy_set_header X-Real-IP \$remote_addr;
  964. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  965. proxy_set_header X-Nginx-Proxy true;
  966. proxy_buffering off;
  967. proxy_redirect off;
  968. }
  969. }"
  970. record_templates[nvcr]="server {
  971. listen 80;
  972. #listen 443 ssl;
  973. server_name nvcr.$nginx_domain;
  974. #ssl_certificate /path/to/your_domain_name.crt;
  975. #ssl_certificate_key /path/to/your_domain_name.key;
  976. #ssl_session_timeout 1d;
  977. #ssl_session_cache shared:SSL:50m;
  978. #ssl_session_tickets off;
  979. #ssl_protocols TLSv1.2 TLSv1.3;
  980. #ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;
  981. #ssl_prefer_server_ciphers on;
  982. #ssl_buffer_size 8k;
  983. proxy_connect_timeout 600;
  984. proxy_send_timeout 600;
  985. proxy_read_timeout 600;
  986. send_timeout 600;
  987. location / {
  988. proxy_pass http://localhost:59000;
  989. proxy_set_header Host \$host;
  990. proxy_set_header X-Real-IP \$remote_addr;
  991. proxy_set_header X-Forwarded-For \$proxy_add_x_forwarded_for;
  992. proxy_set_header X-Nginx-Proxy true;
  993. proxy_buffering off;
  994. proxy_redirect off;
  995. }
  996. }"
  997. > /etc/nginx/conf.d/docker-proxy.conf
  998. for record in "${records_array[@]}"; do
  999. if [[ -n "${record_templates[$record]}" ]]; then
  1000. echo "${record_templates[$record]}" >> /etc/nginx/conf.d/docker-proxy.conf
  1001. fi
  1002. done
  1003. start_attempts=3
  1004. for ((i=1; i<=$start_attempts; i++)); do
  1005. start_nginx
  1006. if pgrep "nginx" > /dev/null; then
  1007. INFO "重新载入配置成功. Nginx服务启动完成"
  1008. break
  1009. else
  1010. if [ $i -eq $start_attempts ]; then
  1011. ERROR "Nginx 在尝试 $start_attempts 后无法启动。请检查配置"
  1012. exit 1
  1013. else
  1014. WARN "第 $i 次启动 Nginx 失败。重试..."
  1015. fi
  1016. fi
  1017. done
  1018. break;;
  1019. n|N )
  1020. WARN "退出配置 Nginx 操作。"
  1021. break;;
  1022. * )
  1023. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}";;
  1024. esac
  1025. done
  1026. }
  1027. function CHECK_DOCKER() {
  1028. status=$(systemctl is-active docker)
  1029. if [ "$status" = "active" ]; then
  1030. INFO "Docker 服务运行正常,请继续..."
  1031. else
  1032. ERROR "Docker 服务未运行,请查看日志报错,定位问题后再次执行脚本!"
  1033. ERROR "-----------服务启动失败,请查看错误日志 ↓↓↓-----------"
  1034. journalctl -u docker.service --no-pager
  1035. ERROR "-----------服务启动失败,请查看错误日志 ↑↑↑-----------"
  1036. exit 1
  1037. fi
  1038. }
  1039. function INSTALL_DOCKER() {
  1040. repo_file="docker-ce.repo"
  1041. url="https://download.docker.com/linux/$repo_type"
  1042. MAX_ATTEMPTS=3
  1043. attempt=0
  1044. success=false
  1045. if [ "$repo_type" = "centos" ] || [ "$repo_type" = "rhel" ]; then
  1046. if ! command -v docker &> /dev/null;then
  1047. while [[ $attempt -lt $MAX_ATTEMPTS ]]; do
  1048. attempt=$((attempt + 1))
  1049. WARN "Docker 未安装,正在进行安装..."
  1050. yum-config-manager --add-repo $url/$repo_file &>/dev/null
  1051. $package_manager -y install docker-ce &>/dev/null
  1052. if [ $? -eq 0 ]; then
  1053. success=true
  1054. break
  1055. fi
  1056. ERROR "Docker 安装失败,正在尝试重新下载 (尝试次数: $attempt)"
  1057. done
  1058. if $success; then
  1059. INFO "Docker 安装成功,版本为:$(docker --version)"
  1060. systemctl restart docker &>/dev/null
  1061. CHECK_DOCKER
  1062. systemctl enable docker &>/dev/null
  1063. else
  1064. ERROR "Docker 安装失败,请尝试手动安装"
  1065. exit 1
  1066. fi
  1067. else
  1068. INFO "Docker 已安装,安装版本为:$(docker --version)"
  1069. systemctl restart docker | grep -E "ERROR|ELIFECYCLE|WARN"
  1070. fi
  1071. elif [ "$repo_type" == "ubuntu" ]; then
  1072. if ! command -v docker &> /dev/null;then
  1073. while [[ $attempt -lt $MAX_ATTEMPTS ]]; do
  1074. attempt=$((attempt + 1))
  1075. WARN "Docker 未安装,正在进行安装..."
  1076. curl -fsSL $url/gpg | sudo apt-key add - &>/dev/null
  1077. add-apt-repository "deb [arch=amd64] $url $(lsb_release -cs) stable" <<< $'\n' &>/dev/null
  1078. $package_manager -y install docker-ce docker-ce-cli containerd.io &>/dev/null
  1079. if [ $? -eq 0 ]; then
  1080. success=true
  1081. break
  1082. fi
  1083. ERROR "Docker 安装失败,正在尝试重新下载 (尝试次数: $attempt)"
  1084. done
  1085. if $success; then
  1086. INFO "Docker 安装成功,版本为:$(docker --version)"
  1087. systemctl restart docker &>/dev/null
  1088. CHECK_DOCKER
  1089. systemctl enable docker &>/dev/null
  1090. else
  1091. ERROR "Docker 安装失败,请尝试手动安装"
  1092. exit 1
  1093. fi
  1094. else
  1095. INFO "Docker 已安装,安装版本为:$(docker --version)"
  1096. systemctl restart docker | grep -E "ERROR|ELIFECYCLE|WARN"
  1097. fi
  1098. elif [ "$repo_type" == "debian" ]; then
  1099. if ! command -v docker &> /dev/null;then
  1100. while [[ $attempt -lt $MAX_ATTEMPTS ]]; do
  1101. attempt=$((attempt + 1))
  1102. WARN "Docker 未安装,正在进行安装..."
  1103. curl -fsSL $url/gpg | sudo apt-key add - &>/dev/null
  1104. add-apt-repository "deb [arch=amd64] $url $(lsb_release -cs) stable" <<< $'\n' &>/dev/null
  1105. $package_manager -y install docker-ce docker-ce-cli containerd.io &>/dev/null
  1106. if [ $? -eq 0 ]; then
  1107. success=true
  1108. break
  1109. fi
  1110. ERROR "Docker 安装失败,正在尝试重新下载 (尝试次数: $attempt)"
  1111. done
  1112. if $success; then
  1113. INFO "Docker 安装成功,版本为:$(docker --version)"
  1114. systemctl restart docker &>/dev/null
  1115. CHECK_DOCKER
  1116. systemctl enable docker &>/dev/null
  1117. else
  1118. ERROR "Docker 安装失败,请尝试手动安装"
  1119. exit 1
  1120. fi
  1121. else
  1122. INFO "Docker 已安装,安装版本为:$(docker --version)"
  1123. systemctl restart docker &>/dev/null
  1124. CHECK_DOCKER
  1125. fi
  1126. else
  1127. ERROR "不支持的操作系统."
  1128. exit 1
  1129. fi
  1130. }
  1131. function INSTALL_COMPOSE() {
  1132. SEPARATOR "安装Docker Compose"
  1133. TAG=`curl -s https://api.github.com/repos/docker/compose/releases/latest | jq -r '.tag_name'`
  1134. url="https://github.com/docker/compose/releases/download/$TAG/docker-compose-$(uname -s)-$(uname -m)"
  1135. MAX_ATTEMPTS=3
  1136. attempt=0
  1137. success=false
  1138. save_path="/usr/local/bin"
  1139. chmod +x $save_path/docker-compose &>/dev/null
  1140. if ! command -v docker-compose &> /dev/null || [ -z "$(docker-compose --version)" ]; then
  1141. WARN "Docker Compose 未安装或安装不完整,正在进行安装..."
  1142. while [ $attempt -lt $MAX_ATTEMPTS ]; do
  1143. attempt=$((attempt + 1))
  1144. wget --continue -q $url -O $save_path/docker-compose
  1145. if [ $? -eq 0 ]; then
  1146. chmod +x $save_path/docker-compose
  1147. version_check=$(docker-compose --version)
  1148. if [ -n "$version_check" ]; then
  1149. success=true
  1150. chmod +x $save_path/docker-compose
  1151. break
  1152. else
  1153. WARN "Docker Compose 下载的文件不完整,正在尝试重新下载 (尝试次数: $attempt)"
  1154. rm -f $save_path/docker-compose
  1155. fi
  1156. fi
  1157. ERROR "Docker Compose 下载失败,正在尝试重新下载 (尝试次数: $attempt)"
  1158. done
  1159. if $success; then
  1160. INFO "Docker Compose 安装成功,版本为:$(docker-compose --version)"
  1161. else
  1162. ERROR "Docker Compose 下载失败,请尝试手动安装docker-compose"
  1163. exit 1
  1164. fi
  1165. else
  1166. chmod +x $save_path/docker-compose
  1167. INFO "Docker Compose 已经安装,版本为:$(docker-compose --version)"
  1168. fi
  1169. }
  1170. function INSTALL_DOCKER_CN() {
  1171. MAX_ATTEMPTS=3
  1172. attempt=0
  1173. success=false
  1174. cpu_arch=$(uname -m)
  1175. save_path="/opt/docker_tgz"
  1176. mkdir -p $save_path
  1177. docker_ver="docker-27.1.2.tgz"
  1178. case $cpu_arch in
  1179. "arm64")
  1180. url="https://raw.gitcode.com/dqzboy/docker/blobs/2ce4d6bc245ab1c4525b3e6f55db7d710681e56f/$docker_ver"
  1181. ;;
  1182. "aarch64")
  1183. url="https://raw.gitcode.com/dqzboy/docker/blobs/2ce4d6bc245ab1c4525b3e6f55db7d710681e56f/$docker_ver"
  1184. ;;
  1185. "x86_64")
  1186. url="https://raw.gitcode.com/dqzboy/docker/blobs/179729785ed2c4a8c99aeb546fb6ac2864d7da5c/$docker_ver"
  1187. ;;
  1188. *)
  1189. ERROR "不支持的CPU架构: $cpu_arch"
  1190. exit 1
  1191. ;;
  1192. esac
  1193. if ! command -v docker &> /dev/null; then
  1194. while [ $attempt -lt $MAX_ATTEMPTS ]; do
  1195. attempt=$((attempt + 1))
  1196. WARN "Docker 未安装,正在进行安装..."
  1197. wget -P "$save_path" "$url" &>/dev/null
  1198. if [ $? -eq 0 ]; then
  1199. success=true
  1200. break
  1201. fi
  1202. ERROR "Docker 安装失败,正在尝试重新下载 (尝试次数: $attempt)"
  1203. done
  1204. if $success; then
  1205. tar -xzf $save_path/$docker_ver -C $save_path
  1206. \cp $save_path/docker/* /usr/bin/ &>/dev/null
  1207. rm -rf $save_path
  1208. INFO "Docker 安装成功,版本为:$(docker --version)"
  1209. cat > /usr/lib/systemd/system/docker.service <<EOF
  1210. [Unit]
  1211. Description=Docker Application Container Engine
  1212. Documentation=https://docs.docker.com
  1213. After=network-online.target firewalld.service
  1214. Wants=network-online.target
  1215. [Service]
  1216. Type=notify
  1217. ExecStart=/usr/bin/dockerd
  1218. ExecReload=/bin/kill -s HUP
  1219. LimitNOFILE=infinity
  1220. LimitNPROC=infinity
  1221. LimitCORE=infinity
  1222. TimeoutStartSec=0
  1223. Delegate=yes
  1224. KillMode=process
  1225. Restart=on-failure
  1226. StartLimitBurst=3
  1227. StartLimitInterval=60s
  1228. [Install]
  1229. WantedBy=multi-user.target
  1230. EOF
  1231. systemctl daemon-reload
  1232. systemctl restart docker &>/dev/null
  1233. CHECK_DOCKER
  1234. systemctl enable docker &>/dev/null
  1235. else
  1236. ERROR "Docker 安装失败,请尝试手动安装"
  1237. exit 1
  1238. fi
  1239. else
  1240. INFO "Docker 已安装,安装版本为:$(docker --version)"
  1241. systemctl restart docker &>/dev/null
  1242. CHECK_DOCKER
  1243. fi
  1244. }
  1245. function INSTALL_COMPOSE_CN() {
  1246. SEPARATOR "安装Docker Compose"
  1247. MAX_ATTEMPTS=3
  1248. attempt=0
  1249. cpu_arch=$(uname -m)
  1250. success=false
  1251. save_path="/usr/local/bin"
  1252. case $cpu_arch in
  1253. "arm64")
  1254. url="https://raw.gitcode.com/dqzboy/docker/blobs/b7be39a4442a103749c769ca48740e8e1a93a16c/docker-compose-linux-aarch64"
  1255. ;;
  1256. "aarch64")
  1257. url="https://raw.gitcode.com/dqzboy/docker/blobs/b7be39a4442a103749c769ca48740e8e1a93a16c/docker-compose-linux-aarch64"
  1258. ;;
  1259. "x86_64")
  1260. url="https://raw.gitcode.com/dqzboy/docker/blobs/df1b7935ced481a20d16141d97e163823ee793f5/docker-compose-linux-x86_64"
  1261. ;;
  1262. *)
  1263. ERROR "不支持的CPU架构: $cpu_arch"
  1264. exit 1
  1265. ;;
  1266. esac
  1267. chmod +x $save_path/docker-compose &>/dev/null
  1268. if ! command -v docker-compose &> /dev/null || [ -z "$(docker-compose --version)" ]; then
  1269. WARN "Docker Compose 未安装或安装不完整,正在进行安装..."
  1270. while [ $attempt -lt $MAX_ATTEMPTS ]; do
  1271. attempt=$((attempt + 1))
  1272. wget --continue -q $url -O $save_path/docker-compose
  1273. if [ $? -eq 0 ]; then
  1274. chmod +x $save_path/docker-compose
  1275. version_check=$(docker-compose --version)
  1276. if [ -n "$version_check" ]; then
  1277. success=true
  1278. chmod +x $save_path/docker-compose
  1279. break
  1280. else
  1281. WARN "Docker Compose 下载的文件不完整,正在尝试重新下载 (尝试次数: $attempt)"
  1282. rm -f $save_path/docker-compose
  1283. fi
  1284. fi
  1285. ERROR "Docker Compose 下载失败,正在尝试重新下载 (尝试次数: $attempt)"
  1286. done
  1287. if $success; then
  1288. INFO "Docker Compose 安装成功,版本为:$(docker-compose --version)"
  1289. else
  1290. ERROR "Docker Compose 下载失败,请尝试手动安装docker-compose"
  1291. exit 1
  1292. fi
  1293. else
  1294. chmod +x $save_path/docker-compose
  1295. INFO "Docker Compose 安装成功,版本为:$(docker-compose --version)"
  1296. fi
  1297. }
  1298. function update_docker_registry_url() {
  1299. local container_name=$1
  1300. if [[ -f "${PROXY_DIR}/${DOCKER_COMPOSE_FILE}" ]]; then
  1301. sed -i "s@- DOCKER_REGISTRY_URL=.*@- DOCKER_REGISTRY_URL=http://${container_name}:5000@g" ${PROXY_DIR}/${DOCKER_COMPOSE_FILE}
  1302. else
  1303. ERROR "文件 ${LIGHT_CYAN}${PROXY_DIR}/${DOCKER_COMPOSE_FILE} ${RESET} ${LIGHT_RED}不存在${RESET},导致容器无法应用新配置"
  1304. exit 1
  1305. fi
  1306. }
  1307. function CONFIG_FILES() {
  1308. while true; do
  1309. read -e -p "$(INFO "安装环境确认 [${LIGHT_GREEN}国外输1${RESET} ${LIGHT_YELLOW}国内输2${RESET}] > ")" install_docker_reg
  1310. case "$install_docker_reg" in
  1311. 1 )
  1312. files=(
  1313. "dockerhub reg-docker-hub ${GITRAW}/config/registry-hub.yml"
  1314. "gcr reg-gcr ${GITRAW}/config/registry-gcr.yml"
  1315. "ghcr reg-ghcr ${GITRAW}/config/registry-ghcr.yml"
  1316. "quay reg-quay ${GITRAW}/config/registry-quay.yml"
  1317. "k8sgcr reg-k8s-gcr ${GITRAW}/config/registry-k8sgcr.yml"
  1318. "k8s reg-k8s ${GITRAW}/config/registry-k8s.yml"
  1319. "mcr reg-mcr ${GITRAW}/config/registry-mcr.yml"
  1320. "elastic reg-elastic ${GITRAW}/config/registry-elastic.yml"
  1321. "nvcr reg-nvcr ${GITRAW}/config/registry-nvcr.yml"
  1322. )
  1323. break;;
  1324. 2 )
  1325. files=(
  1326. "dockerhub reg-docker-hub ${CNGITRAW}/config/registry-hub.yml"
  1327. "gcr reg-gcr ${CNGITRAW}/config/registry-gcr.yml"
  1328. "ghcr reg-ghcr ${CNGITRAW}/config/registry-ghcr.yml"
  1329. "quay reg-quay ${CNGITRAW}/config/registry-quay.yml"
  1330. "k8sgcr reg-k8s-gcr ${CNGITRAW}/config/registry-k8sgcr.yml"
  1331. "k8s reg-k8s ${CNGITRAW}/config/registry-k8s.yml"
  1332. "mcr reg-mcr ${CNGITRAW}/config/registry-mcr.yml"
  1333. "elastic reg-elastic ${CNGITRAW}/config/registry-elastic.yml"
  1334. "nvcr reg-nvcr ${CNGITRAW}/config/registry-nvcr.yml"
  1335. )
  1336. break;;
  1337. * )
  1338. INFO "请输入 ${LIGHT_GREEN}1${RESET} 表示国外 或者 ${LIGHT_YELLOW}2${RESET} 表示大陆";;
  1339. esac
  1340. done
  1341. }
  1342. function DOWN_CONFIG() {
  1343. selected_names=()
  1344. selected_files=()
  1345. selected_containers=()
  1346. REGISTRY_MENU
  1347. read -e -p "$(INFO "输入序号下载对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all下载所有${RESET} > ")" choices_reg
  1348. while true; do
  1349. if [[ "$choices_reg" =~ ^[0-9]+([[:space:]][0-9]+)*$ ]]; then
  1350. valid=true
  1351. for choice in $choices_reg; do
  1352. if ((choice < 0 || choice > 10)); then
  1353. valid=false
  1354. break
  1355. fi
  1356. done
  1357. if $valid; then
  1358. break
  1359. fi
  1360. fi
  1361. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-10 ${RESET}序号"
  1362. read -e -p "$(INFO "输入序号下载对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all下载所有${RESET} > ")" choices_reg
  1363. done
  1364. if [[ "$choices_reg" == "10" ]]; then
  1365. for file in "${files[@]}"; do
  1366. file_name=$(echo "$file" | cut -d' ' -f1)
  1367. container_name=$(echo "$file" | cut -d' ' -f2)
  1368. file_url=$(echo "$file" | cut -d' ' -f3-)
  1369. selected_names+=("$file_name")
  1370. selected_containers+=("$container_name")
  1371. selected_files+=("$file_url")
  1372. wget -NP ${PROXY_DIR}/ $file_url &>/dev/null
  1373. done
  1374. selected_all=true
  1375. elif [[ "$choices_reg" == "0" ]]; then
  1376. WARN "退出下载配置! ${LIGHT_YELLOW}首次安装如果没有配置无法启动Registry服务,只能启动Registry-UI服务${RESET}"
  1377. return
  1378. else
  1379. for choice in ${choices_reg}; do
  1380. if ((choice > 0 && choice < 10)); then
  1381. file_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f1)
  1382. container_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f2)
  1383. file_url=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f3-)
  1384. selected_names+=("$file_name")
  1385. selected_containers+=("$container_name")
  1386. selected_files+=("$file_url")
  1387. wget -NP ${PROXY_DIR}/ $file_url &>/dev/null
  1388. fi
  1389. done
  1390. selected_all=false
  1391. # 非更新配置操作则执行下面步骤
  1392. if [[ "$main_choice" != "4" ]]; then
  1393. first_selected_container=${selected_containers[0]}
  1394. update_docker_registry_url "$first_selected_container"
  1395. fi
  1396. fi
  1397. WARN "${LIGHT_GREEN}>>> 提示:${RESET} ${LIGHT_BLUE}Proxy代理缓存过期时间${RESET} ${MAGENTA}单位:ns、us、ms、s、m、h.默认ns,0禁用缓存过期${RESET}"
  1398. read -e -p "$(INFO "是否要修改缓存时间? ${PROMPT_YES_NO}")" modify_cache
  1399. while [[ "$modify_cache" != "y" && "$modify_cache" != "n" ]]; do
  1400. WARN "无效输入,请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}"
  1401. read -e -p "$(INFO "是否要修改缓存时间? ${PROMPT_YES_NO}")" modify_cache
  1402. done
  1403. if [[ "$modify_cache" == "y" ]]; then
  1404. while true; do
  1405. read -e -p "$(INFO "请输入新的缓存时间值: ")" new_ttl
  1406. for file_url in "${selected_files[@]}"; do
  1407. yml_name=$(basename "$file_url")
  1408. sed -ri "s/ttl: 168h/ttl: ${new_ttl}/g" ${PROXY_DIR}/${yml_name} &>/dev/null
  1409. done
  1410. break
  1411. done
  1412. fi
  1413. }
  1414. # 一键部署调此函数
  1415. function PROXY_HTTP() {
  1416. read -e -p "$(INFO "是否添加代理? ${PROMPT_YES_NO}")" modify_config
  1417. case $modify_config in
  1418. [Yy]* )
  1419. read -e -p "$(INFO "输入代理地址 ${LIGHT_MAGENTA}(eg: host:port)${RESET}: ")" url
  1420. while [[ -z "$url" ]]; do
  1421. WARN "代理${LIGHT_YELLOW}地址不能为空${RESET},请重新输入!"
  1422. read -e -p "$(INFO "输入代理地址 ${LIGHT_MAGENTA}(eg: host:port)${RESET}: ")" url
  1423. done
  1424. sed -i "s@#- http=http://host:port@- http_proxy=http://${url}@g" ${PROXY_DIR}/${DOCKER_COMPOSE_FILE}
  1425. sed -i "s@#- https=http://host:port@- https_proxy=http://${url}@g" ${PROXY_DIR}/${DOCKER_COMPOSE_FILE}
  1426. INFO "你配置代理地址为: ${CYAN}http://${url}${RESET}"
  1427. ;;
  1428. [Nn]* )
  1429. WARN "跳过添加代理配置"
  1430. ;;
  1431. * )
  1432. ERROR "无效的输入。请重新输入${LIGHT_GREEN}Y or N ${RESET}的选项"
  1433. PROXY_HTTP
  1434. ;;
  1435. esac
  1436. }
  1437. # 7) 本机Docker代理,调此函数
  1438. function DOCKER_PROXY_HTTP() {
  1439. WARN "${BOLD}${LIGHT_GREEN}提示:${RESET} ${LIGHT_CYAN}配置本机Docker服务走代理,加速本机Docker镜像下载${RESET}"
  1440. read -e -p "$(INFO "是否添加本机Docker服务代理? ${PROMPT_YES_NO}")" modify_proxy
  1441. case $modify_proxy in
  1442. [Yy]* )
  1443. read -e -p "$(INFO "输入代理地址 ${LIGHT_MAGENTA}(eg: host:port)${RESET}: ")" url
  1444. while [[ -z "$url" ]]; do
  1445. WARN "代理${LIGHT_YELLOW}地址不能为空${RESET},请重新输入。"
  1446. read -e -p "$(INFO "输入代理地址 ${LIGHT_MAGENTA}(eg: host:port)${RESET}: ")" url
  1447. done
  1448. INFO "你配置代理地址为: ${CYAN}http://${url}${RESET}"
  1449. ;;
  1450. [Nn]* )
  1451. WARN "退出本机Docker服务代理配置"
  1452. main_menu
  1453. ;;
  1454. * )
  1455. ERROR "无效的输入。请重新输入${LIGHT_GREEN}Y or N ${RESET}的选项"
  1456. DOCKER_PROXY_HTTP
  1457. ;;
  1458. esac
  1459. }
  1460. function CHECK_DOCKER_PROXY() {
  1461. local url=$1
  1462. local http_proxy=$(docker info 2>/dev/null | grep -i "HTTP Proxy" | awk -F ': ' '{print $2}')
  1463. local https_proxy=$(docker info 2>/dev/null | grep -i "HTTPS Proxy" | awk -F ': ' '{print $2}')
  1464. if [[ "$http_proxy" == "http://$url" && "$https_proxy" == "http://$url" ]]; then
  1465. INFO "Docker 代理${LIGHT_GREEN}配置成功${RESET},当前 HTTP Proxy: ${LIGHT_CYAN}$http_proxy${RESET}, HTTPS Proxy: ${LIGHT_CYAN}$https_proxy${RESET}"
  1466. else
  1467. ERROR "Docker 代理${LIGHT_RED}配置失败${RESET},请检查配置并重新执行配置"
  1468. DOCKER_PROXY_HTTP
  1469. fi
  1470. }
  1471. function ADD_DOCKERD_PROXY() {
  1472. mkdir -p /etc/systemd/system/docker.service.d
  1473. # 设置代理的函数
  1474. set_proxy_config() {
  1475. cat > /etc/systemd/system/docker.service.d/http-proxy.conf <<EOF
  1476. [Service]
  1477. Environment="HTTP_PROXY=http://$url"
  1478. Environment="HTTPS_PROXY=http://$url"
  1479. EOF
  1480. systemctl daemon-reload
  1481. systemctl restart docker &>/dev/null
  1482. CHECK_DOCKER
  1483. CHECK_DOCKER_PROXY "$url"
  1484. }
  1485. # 检查并设置代理配置
  1486. if [ ! -f /etc/systemd/system/docker.service.d/http-proxy.conf ]; then
  1487. # 如果配置文件不存在,直接设置代理
  1488. set_proxy_config
  1489. else
  1490. # 如果配置文件存在,检查是否有相同的代理配置
  1491. if ! grep -q "HTTP_PROXY=http://$url" /etc/systemd/system/docker.service.d/http-proxy.conf || \
  1492. ! grep -q "HTTPS_PROXY=http://$url" /etc/systemd/system/docker.service.d/http-proxy.conf; then
  1493. # 配置文件存在,但没有相同的代理配置,添加新的代理配置
  1494. set_proxy_config
  1495. else
  1496. WARN "已经存在相同的代理配置,${LIGHT_RED}请勿重复配置${RESET}"
  1497. fi
  1498. fi
  1499. }
  1500. function DEL_DOCKERD_PROXY() {
  1501. check_proxy_config() {
  1502. systemctl daemon-reload
  1503. systemctl restart docker &>/dev/null
  1504. CHECK_DOCKER
  1505. }
  1506. WARN "${BOLD}${LIGHT_GREEN}提示:${RESET} ${LIGHT_CYAN}移除本机Docker服务走代理,Docker镜像下载可能会失败!${RESET}"
  1507. read -e -p "$(INFO "是否移除本机Docker服务代理? ${PROMPT_YES_NO}")" del_proxy
  1508. case $del_proxy in
  1509. [Yy]* )
  1510. # 检查并设置代理配置
  1511. if [ ! -f /etc/systemd/system/docker.service.d/http-proxy.conf ]; then
  1512. # 如果配置文件不存在,打印提示
  1513. INFO "本机Docker服务未配置代理"
  1514. else
  1515. # 如果配置文件存在,则进行删除并重启Docker服务
  1516. rm -f /etc/systemd/system/docker.service.d/http-proxy.conf &>/dev/null
  1517. check_proxy_config
  1518. INFO "本机Docker服务代理已移除"
  1519. fi
  1520. ;;
  1521. [Nn]* )
  1522. WARN "退出移除本机Docker服务代理配置"
  1523. main_menu
  1524. ;;
  1525. * )
  1526. ERROR "无效的输入。请重新输入${LIGHT_GREEN}Y or N ${RESET}的选项"
  1527. DOCKER_PROXY_HTTP
  1528. ;;
  1529. esac
  1530. }
  1531. # 一键部署、安装指定容器加速服务时调用START_CONTAINER
  1532. function START_CONTAINER() {
  1533. CHECK_COMPOSE_CMD
  1534. if [ "$modify_config" = "y" ] || [ "$modify_config" = "Y" ]; then
  1535. ADD_DOCKERD_PROXY
  1536. else
  1537. INFO "拉取服务镜像并启动服务中,请稍等..."
  1538. fi
  1539. # DOWN_CONFIG函数执行后判断selected_all变量
  1540. if [ "$selected_all" = true ]; then
  1541. $DOCKER_COMPOSE_CMD up -d --force-recreate
  1542. # 检查命令执行是否成功
  1543. if [ $? -ne 0 ]; then
  1544. ERROR "Docker 容器${LIGHT_RED}启动失败${RESET},请通过查看日志确认启动失败原因"
  1545. exit 1
  1546. fi
  1547. else
  1548. $DOCKER_COMPOSE_CMD up -d "${selected_names[@]}" registry-ui
  1549. # 检查命令执行是否成功
  1550. if [ $? -ne 0 ]; then
  1551. ERROR "Docker 容器${LIGHT_RED}启动失败${RESET},请通过查看日志确认启动失败原因"
  1552. exit 1
  1553. else
  1554. INFO "容器${LIGHT_GREEN}安装完成${RESET},并${LIGHT_GREEN}成功启动${RESET}"
  1555. fi
  1556. fi
  1557. }
  1558. # 使用函数UPDATE_CONFIG时调用RESTART_CONTAINER
  1559. function RESTART_CONTAINER() {
  1560. CHECK_COMPOSE_CMD
  1561. # DOWN_CONFIG函数执行后判断selected_all变量
  1562. if [ "$selected_all" = true ]; then
  1563. $DOCKER_COMPOSE_CMD restart
  1564. # 检查命令执行是否成功
  1565. if [ $? -ne 0 ]; then
  1566. ERROR "Docker 容器启动失败,请通过查看日志确认启动失败原因"
  1567. exit 1
  1568. fi
  1569. else
  1570. $DOCKER_COMPOSE_CMD restart "${selected_names[@]}"
  1571. # 检查命令执行是否成功
  1572. if [ $? -ne 0 ]; then
  1573. ERROR "Docker 容器启动失败,请通过查看日志确认启动失败原因"
  1574. exit 1
  1575. fi
  1576. fi
  1577. }
  1578. function INSTALL_DOCKER_PROXY() {
  1579. SEPARATOR "部署Docker Proxy"
  1580. CONFIG_FILES
  1581. if [[ "$install_docker_reg" == "1" ]]; then
  1582. wget -NP ${PROXY_DIR}/ ${GITRAW}/${DOCKER_COMPOSE_FILE} &>/dev/null
  1583. elif [[ "$install_docker_reg" == "2" ]]; then
  1584. wget -NP ${PROXY_DIR}/ ${CNGITRAW}/${DOCKER_COMPOSE_FILE} &>/dev/null
  1585. fi
  1586. DOWN_CONFIG
  1587. PROXY_HTTP
  1588. START_CONTAINER
  1589. }
  1590. function STOP_REMOVE_CONTAINER() {
  1591. CHECK_COMPOSE_CMD
  1592. if [[ -f "${PROXY_DIR}/${DOCKER_COMPOSE_FILE}" ]]; then
  1593. INFO "停止和移除所有容器"
  1594. $DOCKER_COMPOSE_CMD -f "${PROXY_DIR}/${DOCKER_COMPOSE_FILE}" down --remove-orphans
  1595. else
  1596. WARN "${LIGHT_YELLOW}容器目前未处于运行状态,无需进行删除操作!${RESET}"
  1597. exit 1
  1598. fi
  1599. }
  1600. function UPDATE_CONFIG() {
  1601. while true; do
  1602. read -e -p "$(WARN "是否更新配置,更新前请确保您已备份现有配置,此操作不可逆? ${PROMPT_YES_NO}")" update_conf
  1603. case "$update_conf" in
  1604. y|Y )
  1605. CONFIG_FILES
  1606. DOWN_CONFIG
  1607. RESTART_CONTAINER
  1608. break;;
  1609. n|N )
  1610. WARN "退出配置更新操作。"
  1611. break;;
  1612. * )
  1613. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}";;
  1614. esac
  1615. done
  1616. }
  1617. function REMOVE_NONE_TAG() {
  1618. docker images | grep "^${IMAGE_NAME}.*<none>" | awk '{print $3}' | xargs -r docker rmi
  1619. images=$(docker images ${IMAGE_NAME} --format '{{.Repository}}:{{.Tag}}')
  1620. latest=$(echo "$images" | sort -V | tail -n1)
  1621. for image in $images
  1622. do
  1623. if [ "$image" != "$latest" ];then
  1624. docker rmi $image
  1625. fi
  1626. done
  1627. }
  1628. function PACKAGE() {
  1629. while true; do
  1630. read -e -p "$(INFO "是否执行软件包安装? (${LIGHT_YELLOW}首次部署需安装依赖${RESET}) ${PROMPT_YES_NO}")" choice_package
  1631. case "$choice_package" in
  1632. y|Y )
  1633. INSTALL_PACKAGE
  1634. break;;
  1635. n|N )
  1636. WARN "跳过软件包安装步骤"
  1637. break;;
  1638. * )
  1639. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}";;
  1640. esac
  1641. done
  1642. }
  1643. function INSTALL_WEB() {
  1644. while true; do
  1645. SEPARATOR "安装WEB服务"
  1646. read -e -p "$(INFO "是否安装WEB服务? (用来通过域名方式访问加速服务) ${PROMPT_YES_NO}")" choice_service
  1647. if [[ "$choice_service" =~ ^[YyNn]$ ]]; then
  1648. if [[ "$choice_service" == "Y" || "$choice_service" == "y" ]]; then
  1649. while true; do
  1650. read -e -p "$(INFO "选择安装的WEB服务。安装${LIGHT_CYAN}Caddy可自动开启HTTPS${RESET} [Nginx/Caddy]: ")" web_service
  1651. if [[ "$web_service" =~ ^(nginx|Nginx|caddy|Caddy)$ ]]; then
  1652. if [[ "$web_service" == "nginx" || "$web_service" == "Nginx" ]]; then
  1653. INSTALL_NGINX
  1654. CONFIG_NGINX
  1655. break
  1656. elif [[ "$web_service" == "caddy" || "$web_service" == "Caddy" ]]; then
  1657. INSTALL_CADDY
  1658. CONFIG_CADDY
  1659. break
  1660. fi
  1661. else
  1662. WARN "请输入 ${LIGHT_CYAN}nginx${RESET} 或 ${LIGHT_BLUE}caddy${RESET}"
  1663. fi
  1664. done
  1665. break
  1666. else
  1667. WARN "跳过WEB服务安装步骤"
  1668. break
  1669. fi
  1670. else
  1671. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}"
  1672. fi
  1673. done
  1674. }
  1675. function PROMPT(){
  1676. PUBLIC_IP=$(curl -s https://ifconfig.me)
  1677. ALL_IPS=$(hostname -I)
  1678. INTERNAL_IP=$(echo "$ALL_IPS" | awk '$1!="127.0.0.1" && $1!="::1" && $1!="docker0" {print $1}')
  1679. echo
  1680. INFO "=================感谢您的耐心等待,安装已经完成=================="
  1681. INFO
  1682. INFO "请用浏览器访问 UI 面板: "
  1683. INFO "公网访问地址: ${UNDERLINE}http://$PUBLIC_IP:50000${RESET}"
  1684. INFO "内网访问地址: ${UNDERLINE}http://$INTERNAL_IP:50000${RESET}"
  1685. INFO
  1686. INFO "服务安装路径: ${LIGHT_BLUE}${PROXY_DIR}${RESET}"
  1687. INFO
  1688. INFO "作者博客: https://dqzboy.com"
  1689. INFO "技术交流: https://t.me/dqzboyblog"
  1690. INFO "代码仓库: https://github.com/dqzboy/Docker-Proxy"
  1691. INFO
  1692. INFO "若用云服务器并设域名及证书,需在安全组开放80、443端口;否则开放对应服务监听端口"
  1693. INFO
  1694. INFO "VPS推荐: https://my.racknerd.com/aff.php?aff=12151"
  1695. INFO
  1696. INFO "================================================================"
  1697. }
  1698. function INSTALL_PROXY() {
  1699. ALL_IN_ONE() {
  1700. CHECK_OS
  1701. CHECK_PACKAGE_MANAGER
  1702. CHECK_PKG_MANAGER
  1703. CHECK_COMPOSE_CMD
  1704. CHECKMEM
  1705. CHECKFIRE
  1706. CHECKBBR
  1707. PACKAGE
  1708. INSTALL_WEB
  1709. while true; do
  1710. SEPARATOR "安装Docker"
  1711. read -e -p "$(INFO "安装环境确认 [${LIGHT_GREEN}国外输1${RESET} ${LIGHT_YELLOW}国内输2${RESET}] > ")" deploy_docker
  1712. case "$deploy_docker" in
  1713. 1 )
  1714. INSTALL_DOCKER
  1715. INSTALL_COMPOSE
  1716. break;;
  1717. 2 )
  1718. INSTALL_DOCKER_CN
  1719. INSTALL_COMPOSE_CN
  1720. break;;
  1721. * )
  1722. INFO "请输入 ${LIGHT_GREEN}1${RESET} 表示国外 或者 ${LIGHT_YELLOW}2${RESET} 表示大陆";;
  1723. esac
  1724. done
  1725. INSTALL_DOCKER_PROXY
  1726. PROMPT
  1727. }
  1728. ADD_DOCKER_SERVICE() {
  1729. WARN "提示: 此操作是在你的服务器${LIGHT_CYAN}已经部署对应组件${RESET}后才可执行,否则执行过程将会出现${LIGHT_RED}各种报错!${RESET}"
  1730. INSTALL_DOCKER_PROXY
  1731. }
  1732. SEPARATOR "安装服务"
  1733. echo -e "1) 一键${BOLD}${LIGHT_GREEN}部署所有${RESET}服务"
  1734. echo -e "2) ${BOLD}${LIGHT_CYAN}安装指定${RESET}容器服务"
  1735. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  1736. echo -e "0) ${BOLD}退出脚本${RESET}"
  1737. echo "---------------------------------------------------------------"
  1738. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" proxy_install
  1739. case $proxy_install in
  1740. 1)
  1741. ALL_IN_ONE
  1742. ;;
  1743. 2)
  1744. ADD_DOCKER_SERVICE
  1745. INSTALL_PROXY
  1746. ;;
  1747. 3)
  1748. main_menu
  1749. ;;
  1750. 0)
  1751. exit 1
  1752. ;;
  1753. *)
  1754. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  1755. INSTALL_PROXY
  1756. ;;
  1757. esac
  1758. }
  1759. function REMOVE_CMDUI() {
  1760. CHECK_COMPOSE_CMD
  1761. CMDUI_NAME="hubcmd-ui"
  1762. CMDUI_DIR="${PROXY_DIR}/hubcmdui"
  1763. if [ -d "${CMDUI_DIR}" ]; then
  1764. if [ -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" ]; then
  1765. INFO "停止和移除HubCMD-UI容器"
  1766. $DOCKER_COMPOSE_CMD -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" down --remove-orphans
  1767. rm -rf "${CMDUI_DIR}"
  1768. else
  1769. WARN "${LIGHT_YELLOW}文件${CMDUI_DIR}/${DOCKER_COMPOSE_FILE} 不存在,无需进行删除操作!${RESET}"
  1770. fi
  1771. else
  1772. WARN "${LIGHT_YELLOW}目录 ${CMDUI_DIR} 不存在,无需进行删除操作!${RESET}"
  1773. fi
  1774. docker images | grep "^${CMDUI_IMAGE_NAME}.*<none>" | awk '{print $3}' | xargs -r docker rmi
  1775. images=$(docker images ${CMDUI_IMAGE_NAME} --format '{{.Repository}}:{{.Tag}}')
  1776. latest=$(echo "$images" | sort -V | tail -n1)
  1777. for image in $images; do
  1778. if [ "$image" != "$latest" ]; then
  1779. docker rmi $image
  1780. fi
  1781. done
  1782. # 强制移除所有相关镜像
  1783. docker rmi --force $(docker images -q ${CMDUI_IMAGE_NAME}) &>/dev/null
  1784. }
  1785. function HUBCMDUI() {
  1786. CHECK_COMPOSE_CMD
  1787. CMDUI_NAME="hubcmd-ui"
  1788. CMDUI_DIR="${PROXY_DIR}/hubcmdui"
  1789. CMDUI_PROMPT() {
  1790. PUBLIC_IP=$(curl -s https://ifconfig.me)
  1791. ALL_IPS=$(hostname -I)
  1792. INTERNAL_IP=$(echo "$ALL_IPS" | awk '$1!="127.0.0.1" && $1!="::1" && $1!="docker0" {print $1}')
  1793. echo
  1794. INFO "=================感谢您的耐心等待,安装已经完成=================="
  1795. INFO
  1796. INFO "请用浏览器访问 HubCMD-UI 面板: "
  1797. INFO "公网访问地址: ${UNDERLINE}http://$PUBLIC_IP:30080${RESET}"
  1798. INFO "内网访问地址: ${UNDERLINE}http://$INTERNAL_IP:30080${RESET}"
  1799. INFO
  1800. INFO "后端访问地址: 地址后面跟admin,例: ${UNDERLINE}http://$INTERNAL_IP:30080/admin${RESET}"
  1801. INFO "默认账号密码: ${LIGHT_GREEN}root${RESET}/${LIGHT_CYAN}admin@123${RESET}"
  1802. INFO
  1803. INFO "服务安装路径: ${LIGHT_BLUE}${CMDUI_DIR}${RESET}"
  1804. INFO
  1805. INFO "作者博客: https://dqzboy.com"
  1806. INFO "技术交流: https://t.me/dqzboyblog"
  1807. INFO "代码仓库: https://github.com/dqzboy/Docker-Proxy"
  1808. INFO
  1809. INFO "VPS推荐: https://my.racknerd.com/aff.php?aff=12151"
  1810. INFO
  1811. INFO "================================================================"
  1812. }
  1813. INSTALL_HUBCMDUI() {
  1814. mkdir -p ${CMDUI_DIR}
  1815. INFO "正在安装HubCMD-UI服务,请稍等!安装路径 ${LIGHT_CYAN}${CMDUI_DIR}${RESET}"
  1816. if [[ -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" ]]; then
  1817. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${CMDUI_NAME}$"; then
  1818. INFO "${CMDUI_NAME} 已经安装并启动,无需重复执行安装!"
  1819. else
  1820. $DOCKER_COMPOSE_CMD -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" up -d
  1821. fi
  1822. else
  1823. wget -NP ${CMDUI_DIR}/ ${CMDUI_COMPOSE_FILE} &>/dev/null
  1824. if [ $? -ne 0 ]; then
  1825. WARN "下载${LIGHT_YELLOW}docker-compose.yaml 文件失败${RESET},请稍后重试!"
  1826. HUBCMDUI
  1827. fi
  1828. $DOCKER_COMPOSE_CMD -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" up -d
  1829. if [ $? -eq 0 ]; then
  1830. CMDUI_PROMPT
  1831. exit 1
  1832. else
  1833. WARN "服务安装失败,请稍后重试!"
  1834. HUBCMDUI
  1835. fi
  1836. fi
  1837. }
  1838. UPDATE_HUBCMDUI() {
  1839. if [ -d "${CMDUI_DIR}" ]; then
  1840. if [ -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" ]; then
  1841. INFO "正在更新HubCMD-UI容器"
  1842. $DOCKER_COMPOSE_CMD -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" pull
  1843. if [ $? -ne 0 ]; then
  1844. WARN "HubCMD-UI ${LIGHT_YELLOW}镜像拉取失败${RESET},请稍后重试!"
  1845. HUBCMDUI
  1846. fi
  1847. $DOCKER_COMPOSE_CMD -f "${CMDUI_DIR}/${DOCKER_COMPOSE_FILE}" up -d --force-recreate
  1848. if [ $? -ne 0 ]; then
  1849. WARN "HubCMD-UI ${LIGHT_YELLOW}服务启动失败${RESET},请稍后重试!"
  1850. HUBCMDUI
  1851. else
  1852. INFO "HubCMD-UI ${LIGHT_GREEN}服务更新并启动完成${RESET}"
  1853. fi
  1854. else
  1855. WARN "${LIGHT_YELLOW}文件${CMDUI_DIR}/${DOCKER_COMPOSE_FILE} 不存在,无法进行更新操作!${RESET}"
  1856. fi
  1857. else
  1858. WARN "${LIGHT_YELLOW}目录 ${CMDUI_DIR} 不存在,无法进行更新操作!${RESET}"
  1859. fi
  1860. }
  1861. UNINSTALL_HUBCMDUI() {
  1862. WARN "${LIGHT_RED}注意:${RESET} ${LIGHT_YELLOW}请执行删除之前确定是否需要备份配置文件${RESET}"
  1863. while true; do
  1864. read -e -p "$(INFO "本人${LIGHT_RED}已知晓后果,确认删除${RESET}服务? ${PROMPT_YES_NO}")" delcmdui
  1865. case "$delcmdui" in
  1866. y|Y )
  1867. REMOVE_CMDUI
  1868. break;;
  1869. n|N )
  1870. WARN "退出执行卸载服务"
  1871. break;;
  1872. * )
  1873. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}";;
  1874. esac
  1875. done
  1876. }
  1877. SEPARATOR "HubCMD-UI管理"
  1878. echo -e "1) ${BOLD}${LIGHT_GREEN}安装${RESET}HubCMD-UI"
  1879. echo -e "2) ${BOLD}${LIGHT_YELLOW}卸载${RESET}HubCMD-UI"
  1880. echo -e "3) ${BOLD}${LIGHT_CYAN}更新${RESET}HubCMD-UI"
  1881. echo -e "4) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  1882. echo -e "0) ${BOLD}退出脚本${RESET}"
  1883. echo "---------------------------------------------------------------"
  1884. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" cmdui_choice
  1885. case $cmdui_choice in
  1886. 1)
  1887. INSTALL_HUBCMDUI
  1888. HUBCMDUI
  1889. ;;
  1890. 2)
  1891. UNINSTALL_HUBCMDUI
  1892. HUBCMDUI
  1893. ;;
  1894. 3)
  1895. UPDATE_HUBCMDUI
  1896. HUBCMDUI
  1897. ;;
  1898. 4)
  1899. main_menu
  1900. ;;
  1901. 0)
  1902. exit 1
  1903. ;;
  1904. *)
  1905. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-4${RESET}的选项."
  1906. HUBCMDUI
  1907. ;;
  1908. esac
  1909. }
  1910. function COMP_INST() {
  1911. CHECK_COMPOSE_CMD
  1912. SEPARATOR "安装组件"
  1913. echo -e "1) ${BOLD}安装${LIGHT_GREEN}环境依赖${RESET}"
  1914. echo -e "2) ${BOLD}安装${LIGHT_CYAN}Docker${RESET}"
  1915. echo -e "3) ${BOLD}安装${LIGHT_MAGENTA}Compose${RESET}"
  1916. echo -e "4) ${BOLD}安装${GREEN}Nginx${RESET}"
  1917. echo -e "5) ${BOLD}安装${LIGHT_BLUE}Caddy${RESET}"
  1918. echo -e "6) ${BOLD}配置${LIGHT_YELLOW}Nginx${RESET}"
  1919. echo -e "7) ${BOLD}配置${CYAN}Caddy${RESET}"
  1920. echo -e "8) ${BOLD}安装${BLUE}HubCMD-UI${RESET}"
  1921. echo -e "9) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  1922. echo -e "0) ${BOLD}退出脚本${RESET}"
  1923. echo "---------------------------------------------------------------"
  1924. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" comp_choice
  1925. case $comp_choice in
  1926. 1)
  1927. CHECK_OS
  1928. CHECK_PACKAGE_MANAGER
  1929. CHECK_PKG_MANAGER
  1930. CHECKMEM
  1931. PACKAGE
  1932. COMP_INST
  1933. ;;
  1934. 2)
  1935. CHECK_OS
  1936. CHECK_PACKAGE_MANAGER
  1937. CHECK_PKG_MANAGER
  1938. while true; do
  1939. SEPARATOR "安装Docker"
  1940. read -e -p "$(INFO "安装环境确认 [${LIGHT_GREEN}国外输1${RESET} ${LIGHT_YELLOW}国内输2${RESET}] > ")" deploy_docker
  1941. case "$deploy_docker" in
  1942. 1 )
  1943. INSTALL_DOCKER
  1944. break;;
  1945. 2 )
  1946. INSTALL_DOCKER_CN
  1947. break;;
  1948. * )
  1949. INFO "请输入 ${LIGHT_GREEN}1${RESET} 表示国外 或者 ${LIGHT_YELLOW}2${RESET} 表示大陆";;
  1950. esac
  1951. done
  1952. COMP_INST
  1953. ;;
  1954. 3)
  1955. CHECK_OS
  1956. CHECK_PACKAGE_MANAGER
  1957. CHECK_PKG_MANAGER
  1958. while true; do
  1959. read -e -p "$(INFO "安装环境确认 [${LIGHT_GREEN}国外输1${RESET} ${LIGHT_YELLOW}国内输2${RESET}] > ")" deploy_compose
  1960. case "$deploy_compose" in
  1961. 1 )
  1962. INSTALL_COMPOSE
  1963. break;;
  1964. 2 )
  1965. INSTALL_COMPOSE_CN
  1966. break;;
  1967. * )
  1968. INFO "请输入 ${LIGHT_GREEN}1${RESET} 表示国外 或者 ${LIGHT_YELLOW}2${RESET} 表示大陆";;
  1969. esac
  1970. done
  1971. COMP_INST
  1972. ;;
  1973. 4)
  1974. CHECK_OS
  1975. CHECK_PACKAGE_MANAGER
  1976. CHECK_PKG_MANAGER
  1977. INSTALL_NGINX
  1978. COMP_INST
  1979. ;;
  1980. 5)
  1981. CHECK_OS
  1982. CHECK_PACKAGE_MANAGER
  1983. CHECK_PKG_MANAGER
  1984. INSTALL_CADDY
  1985. COMP_INST
  1986. ;;
  1987. 6)
  1988. CONFIG_NGINX
  1989. COMP_INST
  1990. ;;
  1991. 7)
  1992. CONFIG_CADDY
  1993. COMP_INST
  1994. ;;
  1995. 8)
  1996. HUBCMDUI
  1997. COMP_INST
  1998. ;;
  1999. 9)
  2000. main_menu
  2001. ;;
  2002. 0)
  2003. exit 1
  2004. ;;
  2005. *)
  2006. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-8${RESET}的选项."
  2007. COMP_INST
  2008. ;;
  2009. esac
  2010. }
  2011. function SVC_MGMT() {
  2012. CHECK_COMPOSE_CMD
  2013. RESTART_SERVICE() {
  2014. CONTAINER_SERVICES
  2015. selected_services=()
  2016. WARN "重启服务请在${LIGHT_GREEN}${DOCKER_COMPOSE_FILE}${RESET}文件存储目录下执行脚本.默认安装路径: ${LIGHT_BLUE}${PROXY_DIR}${RESET}"
  2017. REGISTRY_MENU
  2018. read -e -p "$(INFO "输入序号选择对应服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all表示所有${RESET} > ")" restart_service
  2019. while true; do
  2020. if [[ "$restart_service" =~ ^[0-9]+([[:space:]][0-9]+)*$ ]]; then
  2021. valid=true
  2022. for choice in $restart_service; do
  2023. if ((choice < 0 || choice > 10)); then
  2024. valid=false
  2025. break
  2026. fi
  2027. done
  2028. if $valid; then
  2029. break
  2030. fi
  2031. fi
  2032. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-10 ${RESET}序号"
  2033. read -e -p "$(INFO "输入序号选择对应服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all表示所有${RESET} > ")" restart_service
  2034. done
  2035. if [[ "$restart_service" == "10" ]]; then
  2036. for service_name in "${services[@]}"; do
  2037. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2038. selected_services+=("$service_name")
  2039. else
  2040. WARN "服务 ${service_name}未运行,跳过重启。"
  2041. fi
  2042. done
  2043. if [ ${#selected_services[@]} -eq 0 ]; then
  2044. WARN "选择的服务未运行,无需进行重启"
  2045. else
  2046. INFO "更新的服务: ${selected_services[*]}"
  2047. fi
  2048. elif [[ "$restart_service" == "0" ]]; then
  2049. WARN "退出重启服务!"
  2050. SVC_MGMT
  2051. else
  2052. for choice in ${restart_service}; do
  2053. if ((choice > 0 && choice < 10)); then
  2054. service_name="${services[$((choice -1))]}"
  2055. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2056. selected_services+=("$service_name")
  2057. else
  2058. WARN "服务 ${service_name} 未运行,跳过重启。"
  2059. fi
  2060. else
  2061. ERROR "无效的选择: $choice. 请重新${LIGHT_GREEN}选择0-9${RESET}的选项"
  2062. RESTART_SERVICE # 选择无效重新调用当前函数进行选择
  2063. fi
  2064. done
  2065. if [ ${#selected_services[@]} -eq 0 ]; then
  2066. WARN "选择的服务未运行,无需进行重启"
  2067. else
  2068. INFO "更新的服务: ${selected_services[*]}"
  2069. fi
  2070. fi
  2071. }
  2072. UPDATE_SERVICE() {
  2073. CONTAINER_SERVICES
  2074. selected_services=()
  2075. WARN "更新服务请在${LIGHT_GREEN}${DOCKER_COMPOSE_FILE}${RESET}文件存储目录下执行脚本.默认安装路径: ${LIGHT_BLUE}${PROXY_DIR}${RESET}"
  2076. REGISTRY_MENU
  2077. read -e -p "$(INFO "输入序号选择对应服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all表示所有${RESET} > ")" update_service
  2078. while true; do
  2079. if [[ "$update_service" =~ ^[0-9]+([[:space:]][0-9]+)*$ ]]; then
  2080. valid=true
  2081. for choice in $update_service; do
  2082. if ((choice < 0 || choice > 10)); then
  2083. valid=false
  2084. break
  2085. fi
  2086. done
  2087. if $valid; then
  2088. break
  2089. fi
  2090. fi
  2091. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-10 ${RESET}序号"
  2092. read -e -p "$(INFO "输入序号选择对应服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all表示所有${RESET} > ")" update_service
  2093. done
  2094. if [[ "$update_service" == "10" ]]; then
  2095. for service_name in "${services[@]}"; do
  2096. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2097. selected_services+=("$service_name")
  2098. else
  2099. WARN "服务 ${service_name}未运行,跳过更新。"
  2100. fi
  2101. done
  2102. if [ ${#selected_services[@]} -eq 0 ]; then
  2103. WARN "选择的服务未运行,无法进行更新"
  2104. else
  2105. INFO "更新的服务: ${selected_services[*]}"
  2106. fi
  2107. elif [[ "$update_service" == "0" ]]; then
  2108. WARN "退出更新服务!"
  2109. SVC_MGMT
  2110. else
  2111. for choice in ${update_service}; do
  2112. if ((choice > 0 && choice < 10)); then
  2113. service_name="${services[$((choice -1))]}"
  2114. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2115. selected_services+=("$service_name")
  2116. else
  2117. WARN "服务 ${service_name} 未运行,跳过更新。"
  2118. fi
  2119. else
  2120. ERROR "无效的选择: $choice. 请重新${LIGHT_GREEN}选择0-9${RESET}的选项"
  2121. UPDATE_SERVICE # 选择无效重新调用当前函数进行选择
  2122. fi
  2123. done
  2124. if [ ${#selected_services[@]} -eq 0 ]; then
  2125. WARN "选择的服务未运行,无法进行更新"
  2126. else
  2127. INFO "更新的服务: ${selected_services[*]}"
  2128. fi
  2129. fi
  2130. }
  2131. CONTAIENR_LOGS() {
  2132. CONTAINER_SERVICES
  2133. selected_services=()
  2134. REGISTRY_SER_MENU
  2135. read -e -p "$(INFO "输入序号选择对应服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项. ${LIGHT_CYAN}all选择所有${RESET} > ")" restart_service
  2136. if [[ "$restart_service" == "0" ]]; then
  2137. WARN "退出查看容器服务日志操作!"
  2138. SVC_MGMT
  2139. else
  2140. for choice in ${restart_service}; do
  2141. if [[ $choice =~ ^[0-9]+$ ]] && ((choice >0 && choice <= ${#services[@]})); then
  2142. service_name="${services[$((choice -1))]}"
  2143. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2144. selected_services+=("$service_name")
  2145. else
  2146. WARN "服务 ${service_name} 未运行,无法查看容器日志。"
  2147. fi
  2148. else
  2149. ERROR "无效的选择: $choice. 请重新${LIGHT_GREEN}选择0-9${RESET}的选项"
  2150. CONTAIENR_LOGS # 选择无效重新调用当前函数进行选择
  2151. fi
  2152. done
  2153. if [ ${#selected_services[@]} -eq 0 ]; then
  2154. WARN "选择的服务未运行,无法查看日志"
  2155. else
  2156. INFO "查看日志的服务: ${selected_services[*]}"
  2157. fi
  2158. fi
  2159. }
  2160. MODIFY_SERVICE_TTL_CONFIG() {
  2161. selected_services=()
  2162. selected_files=()
  2163. existing_files=()
  2164. non_existing_files=()
  2165. REGISTRY_FILES
  2166. while true; do
  2167. REGISTRY_SER_MENU
  2168. read -e -p "$(INFO "输入序号修改服务对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" ttl_service
  2169. if [[ "$ttl_service" == "0" ]]; then
  2170. WARN "退出修改容器服务缓存配置!"
  2171. SVC_MGMT
  2172. elif [[ "$ttl_service" =~ ^([1-8]+[[:space:]]*)+$ ]]; then
  2173. break
  2174. else
  2175. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2176. fi
  2177. done
  2178. for choice in ${ttl_service}; do
  2179. file_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f2)
  2180. service_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f1)
  2181. selected_files+=("$file_name")
  2182. if [ -f "${PROXY_DIR}/${file_name}" ]; then
  2183. existing_files+=("$file_name")
  2184. selected_services+=("$service_name")
  2185. else
  2186. non_existing_files+=("$file_name")
  2187. fi
  2188. if ! $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2189. WARN "服务 ${service_name} 未运行。"
  2190. fi
  2191. done
  2192. if [ ${#existing_files[@]} -gt 0 ]; then
  2193. INFO "${GREEN}存在的配置文件:${RESET} ${existing_files[*]}${RESET}"
  2194. fi
  2195. if [ ${#non_existing_files[@]} -gt 0 ]; then
  2196. WARN "${RED}不存在的配置文件:${RESET} ${non_existing_files[*]}"
  2197. fi
  2198. if [ ${#existing_files[@]} -gt 0 ]; then
  2199. WARN "${LIGHT_GREEN}>>> 提示:${RESET} ${LIGHT_BLUE}Proxy代理缓存过期时间${RESET} ${MAGENTA}单位:ns、us、ms、s、m、h.默认ns,0禁用缓存过期${RESET}"
  2200. read -e -p "$(INFO "是否要修改缓存时间? ${PROMPT_YES_NO}")" modify_cache
  2201. while [[ "$modify_cache" != "y" && "$modify_cache" != "n" ]]; do
  2202. WARN "无效输入,请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}"
  2203. read -e -p "$(INFO "是否要修改缓存时间? ${PROMPT_YES_NO}")" modify_cache
  2204. done
  2205. if [[ "$modify_cache" == "y" ]]; then
  2206. while true; do
  2207. read -e -p "$(INFO "请输入新的缓存时间值: ")" new_ttl
  2208. for file_url in "${existing_files[@]}"; do
  2209. yml_name=$(basename "$file_url")
  2210. WARN "${YELLOW}正在修改配置文件: ${PROXY_DIR}/${yml_name}${RESET}"
  2211. sed -i "s/ttl: .*/ttl: ${new_ttl}/g" "${PROXY_DIR}/${yml_name}" &>/dev/null
  2212. INFO "${GREEN}配置文件 ${yml_name} 修改完成,代理缓存过期时间已设置为: ${new_ttl}${RESET}"
  2213. done
  2214. break
  2215. done
  2216. fi
  2217. else
  2218. WARN "未选择有效的配置文件进行修改。"
  2219. fi
  2220. }
  2221. ### 启动新容器
  2222. START_NEW_SERVER_DOWN_CONFIG() {
  2223. selected_names=()
  2224. selected_files=()
  2225. selected_containers=()
  2226. REGISTRY_SER_MENU
  2227. read -e -p "$(INFO "输入序号下载对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" choices_newser
  2228. while [[ ! "$choices_newser" =~ ^([0-9]+[[:space:]]*)+$ ]]; do
  2229. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2230. read -e -p "$(INFO "输入序号下载对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" choices_newser
  2231. done
  2232. if [[ "$choices_newser" == "0" ]]; then
  2233. WARN "退出下载配置! ${LIGHT_YELLOW}没有配置将无法启动服务!!!${RESET}"
  2234. SVC_MGMT
  2235. else
  2236. for choice in ${choices_newser}; do
  2237. if [[ $choice =~ ^[0-9]+$ ]] && ((choice > 0 && choice <= ${#files[@]})); then
  2238. file_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f1)
  2239. container_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f2)
  2240. file_url=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f3-)
  2241. selected_names+=("$file_name")
  2242. selected_containers+=("$container_name")
  2243. selected_files+=("$file_url")
  2244. wget -NP ${PROXY_DIR}/ $file_url &>/dev/null
  2245. else
  2246. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2247. fi
  2248. done
  2249. fi
  2250. WARN "${LIGHT_GREEN}>>> 提示:${RESET} ${LIGHT_BLUE}Proxy代理缓存过期时间${RESET} ${MAGENTA}单位:ns、us、ms、s、m、h.默认ns,0禁用缓存过期${RESET}"
  2251. read -e -p "$(INFO "是否要修改缓存时间? ${PROMPT_YES_NO}")" modify_cache
  2252. while [[ "$modify_cache" != "y" && "$modify_cache" != "n" ]]; do
  2253. WARN "无效输入,请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}"
  2254. read -e -p "$(INFO "是否要修改缓存时间? ${PROMPT_YES_NO}")" modify_cache
  2255. done
  2256. if [[ "$modify_cache" == "y" ]]; then
  2257. while true; do
  2258. read -e -p "$(INFO "请输入新的缓存时间值: ")" new_ttl
  2259. for file_url in "${selected_files[@]}"; do
  2260. yml_name=$(basename "$file_url")
  2261. sed -ri "s/ttl: 168h/ttl: ${new_ttl}/g" ${PROXY_DIR}/${yml_name} &>/dev/null
  2262. done
  2263. break
  2264. done
  2265. fi
  2266. }
  2267. START_NEW_DOCKER_SERVICE() {
  2268. if [ -d "${PROXY_DIR}" ]; then
  2269. if [ -f "${PROXY_DIR}/${DOCKER_COMPOSE_FILE}" ]; then
  2270. CONFIG_FILES
  2271. START_NEW_SERVER_DOWN_CONFIG
  2272. PROXY_HTTP
  2273. INFO "正在启动新的容器服务,请稍等..."
  2274. $DOCKER_COMPOSE_CMD -f "${PROXY_DIR}/${DOCKER_COMPOSE_FILE}" up -d "${selected_names[@]}"
  2275. if [ $? -ne 0 ]; then
  2276. WARN "${selected_names[*]} ${LIGHT_YELLOW}服务启动失败${RESET},请排查!"
  2277. else
  2278. INFO "${selected_names[*]} ${LIGHT_GREEN}服务启动完成${RESET}"
  2279. fi
  2280. else
  2281. WARN "${LIGHT_YELLOW}文件${PROXY_DIR}/${DOCKER_COMPOSE_FILE} 不存在,无法启动新的容器!${RESET}"
  2282. fi
  2283. else
  2284. WARN "${LIGHT_YELLOW}目录 ${PROXY_DIR} 不存在,无法启动新的容器!${RESET}"
  2285. fi
  2286. }
  2287. ### 启动新容器 END
  2288. SEPARATOR "服务管理"
  2289. echo -e "1) ${BOLD}${LIGHT_GREEN}重启${RESET}服务"
  2290. echo -e "2) ${BOLD}${LIGHT_CYAN}更新${RESET}服务"
  2291. echo -e "3) ${BOLD}${LIGHT_MAGENTA}查看${RESET}日志"
  2292. echo -e "4) ${BOLD}${LIGHT_BLUE}缓存${RESET}时效"
  2293. echo -e "5) ${BOLD}启动${CYAN}新服务${RESET}"
  2294. echo -e "6) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  2295. echo -e "0) ${BOLD}退出脚本${RESET}"
  2296. echo "---------------------------------------------------------------"
  2297. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" ser_choice
  2298. case $ser_choice in
  2299. 1)
  2300. RESTART_SERVICE
  2301. if [ ${#selected_services[@]} -eq 0 ]; then
  2302. ERROR "没有需要重启的服务,请重新选择"
  2303. RESTART_SERVICE
  2304. else
  2305. $DOCKER_COMPOSE_CMD stop ${selected_services[*]}
  2306. $DOCKER_COMPOSE_CMD up -d --force-recreate ${selected_services[*]}
  2307. fi
  2308. SVC_MGMT
  2309. ;;
  2310. 2)
  2311. UPDATE_SERVICE
  2312. if [ ${#selected_services[@]} -eq 0 ]; then
  2313. ERROR "没有需要更新的服务,请重新选择"
  2314. UPDATE_SERVICE
  2315. else
  2316. $DOCKER_COMPOSE_CMD pull ${selected_services[*]}
  2317. $DOCKER_COMPOSE_CMD up -d --force-recreate ${selected_services[*]}
  2318. fi
  2319. SVC_MGMT
  2320. ;;
  2321. 3)
  2322. CONTAIENR_LOGS
  2323. if [ ${#selected_services[@]} -eq 0 ]; then
  2324. ERROR "没有需要查看的服务,请重新选择"
  2325. CONTAIENR_LOGS
  2326. else
  2327. # 查看最近30条日志
  2328. $DOCKER_COMPOSE_CMD logs --tail=30 ${selected_services[*]}
  2329. fi
  2330. SVC_MGMT
  2331. ;;
  2332. 4)
  2333. MODIFY_SERVICE_TTL_CONFIG
  2334. if [ ${#selected_services[@]} -eq 0 ]; then
  2335. ERROR "修改的服务未运行,请重新选择"
  2336. MODIFY_SERVICE_TTL_CONFIG
  2337. else
  2338. $DOCKER_COMPOSE_CMD restart ${selected_services[*]}
  2339. fi
  2340. SVC_MGMT
  2341. ;;
  2342. 5)
  2343. START_NEW_DOCKER_SERVICE
  2344. SVC_MGMT
  2345. ;;
  2346. 6)
  2347. main_menu
  2348. ;;
  2349. 0)
  2350. exit 1
  2351. ;;
  2352. *)
  2353. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-5${RESET}的选项."
  2354. SVC_MGMT
  2355. ;;
  2356. esac
  2357. }
  2358. function ADD_SYS_CMD() {
  2359. MAX_ATTEMPTS=3
  2360. attempt=0
  2361. success=false
  2362. TARGET_PATH="/usr/bin/hub"
  2363. INSTALL_ENV() {
  2364. while true; do
  2365. read -e -p "$(INFO "安装环境确认 [${LIGHT_GREEN}国外输1${RESET} ${LIGHT_YELLOW}国内输2${RESET}] > ")" sys_cmd
  2366. case "$sys_cmd" in
  2367. 1 )
  2368. DOWNLOAD_URL="https://raw.githubusercontent.com/dqzboy/Docker-Proxy/main/install/DockerProxy_Install.sh"
  2369. break;;
  2370. 2 )
  2371. DOWNLOAD_URL="https://cdn.jsdelivr.net/gh/dqzboy/Docker-Proxy/install/DockerProxy_Install.sh"
  2372. break;;
  2373. * )
  2374. INFO "请输入 ${LIGHT_GREEN}1${RESET} 表示国外 或者 ${LIGHT_YELLOW}2${RESET} 表示大陆";;
  2375. esac
  2376. done
  2377. }
  2378. INSTALL_OR_UPDATE_CMD() {
  2379. local action=$1
  2380. while [[ $attempt -lt $MAX_ATTEMPTS ]]; do
  2381. attempt=$((attempt + 1))
  2382. if [[ "$action" == "安装" ]]; then
  2383. if command -v hub &> /dev/null; then
  2384. INFO "系统命令已存在,无需安装。"
  2385. success=true
  2386. break
  2387. fi
  2388. WARN "正在安装脚本中,请稍等..."
  2389. else
  2390. WARN "正在进行脚本更新,请稍等..."
  2391. if [ -f "$TARGET_PATH" ]; then
  2392. rm -f "$TARGET_PATH" &>/dev/null
  2393. fi
  2394. fi
  2395. wget -q -O "$TARGET_PATH" "$DOWNLOAD_URL" &>/dev/null
  2396. if [ $? -eq 0 ]; then
  2397. success=true
  2398. chmod +x "$TARGET_PATH"
  2399. break
  2400. fi
  2401. ERROR "${action}脚本${RED}失败${RESET},正在尝试重新${action} (尝试次数: $attempt)"
  2402. done
  2403. if $success; then
  2404. INFO "${action}脚本${GREEN}成功${RESET},命令行输入 ${LIGHT_GREEN}hub${RESET} 运行"
  2405. else
  2406. ERROR "设置系统命令失败"
  2407. exit 1
  2408. fi
  2409. }
  2410. SEPARATOR "设置脚本为系统命令"
  2411. echo -e "1) ${BOLD}安装${LIGHT_GREEN}系统命令${RESET}"
  2412. echo -e "2) ${BOLD}更新${LIGHT_CYAN}系统命令${RESET}"
  2413. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  2414. echo -e "0) ${BOLD}退出脚本${RESET}"
  2415. echo "---------------------------------------------------------------"
  2416. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" cmd_choice
  2417. case $cmd_choice in
  2418. 1)
  2419. INSTALL_ENV
  2420. INSTALL_OR_UPDATE_CMD "安装"
  2421. ;;
  2422. 2)
  2423. INSTALL_ENV
  2424. INSTALL_OR_UPDATE_CMD "更新"
  2425. ;;
  2426. 3)
  2427. main_menu
  2428. ;;
  2429. 0)
  2430. exit 1
  2431. ;;
  2432. *)
  2433. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  2434. ADD_SYS_CMD
  2435. ;;
  2436. esac
  2437. }
  2438. function UNI_DOCKER_SERVICE() {
  2439. CHECK_COMPOSE_CMD
  2440. RM_SERVICE() {
  2441. selected_containers=()
  2442. REGISTRY_FILES
  2443. REGISTRY_SER_MENU
  2444. read -e -p "$(INFO "输入序号删除服务和对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" rm_service
  2445. while [[ ! "$rm_service" =~ ^([0-9]+[[:space:]]*)+$ ]]; do
  2446. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2447. read -e -p "$(INFO "输入序号删除服务和对应配置文件,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" rm_service
  2448. done
  2449. if [[ "$rm_service" == "0" ]]; then
  2450. WARN "退出删除容器服务操作!"
  2451. return
  2452. else
  2453. selected_services=()
  2454. for choice in ${rm_service}; do
  2455. if [[ $choice =~ ^[0-9]+$ ]] && ((choice > 0 && choice <= ${#files[@]})); then
  2456. file_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f2)
  2457. service_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f1)
  2458. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2459. selected_services+=("$service_name")
  2460. else
  2461. WARN "服务 ${LIGHT_MAGENTA}${service_name} 未运行${RESET},但将尝试删除相关文件。"
  2462. fi
  2463. if [ -f "${PROXY_DIR}/${file_name}" ]; then
  2464. rm -f "${PROXY_DIR}/${file_name}"
  2465. INFO "配置文件 ${LIGHT_CYAN}${file_name}${RESET} ${LIGHT_GREEN}已被删除${RESET}"
  2466. else
  2467. WARN "配置文件 ${LIGHT_CYAN}${file_name}${RESET} 不存在,${LIGHT_YELLOW}无需删除${RESET}"
  2468. fi
  2469. else
  2470. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2471. UNI_DOCKER_SERVICE
  2472. return
  2473. fi
  2474. done
  2475. # 一次性删除所有选中的服务
  2476. if [ ${#selected_services[@]} -gt 0 ]; then
  2477. INFO "删除的服务: ${LIGHT_RED}${selected_services[*]}${RESET}"
  2478. $DOCKER_COMPOSE_CMD down ${selected_services[*]}
  2479. fi
  2480. fi
  2481. }
  2482. RM_ALLSERVICE() {
  2483. STOP_REMOVE_CONTAINER
  2484. REMOVE_NONE_TAG
  2485. REMOVE_CMDUI
  2486. docker rmi --force $(docker images -q ${IMAGE_NAME}) &>/dev/null
  2487. docker rmi --force $(docker images -q ${UI_IMAGE_NAME}) &>/dev/null
  2488. if [ -d "${PROXY_DIR}" ]; then
  2489. rm -rf "${PROXY_DIR}" &>/dev/null
  2490. fi
  2491. if [ -f "/usr/bin/hub" ]; then
  2492. rm -f /usr/bin/hub &>/dev/null
  2493. fi
  2494. INFO "${LIGHT_YELLOW}感谢您的使用,Docker-Proxy服务已卸载。欢迎您再次使用!${RESET}"
  2495. SEPARATOR "DONE"
  2496. }
  2497. CONFIREM_ACTION() {
  2498. local action_name=$1
  2499. local action_function=$2
  2500. WARN "${LIGHT_RED}注意:${RESET} ${LIGHT_YELLOW}卸载服务会一同将本地的配置和对应服务删除,请执行删除之前确定是否需要备份本地的配置文件${RESET}"
  2501. while true; do
  2502. read -e -p "$(INFO "本人${LIGHT_RED}已知晓后果,确认${action_name}${RESET}服务? ${PROMPT_YES_NO}")" uniservice
  2503. case "$uniservice" in
  2504. y|Y )
  2505. $action_function
  2506. break;;
  2507. n|N )
  2508. WARN "退出${action_name}服务."
  2509. break;;
  2510. * )
  2511. INFO "请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}";;
  2512. esac
  2513. done
  2514. }
  2515. SEPARATOR "卸载服务"
  2516. echo -e "1) ${BOLD}卸载${LIGHT_YELLOW}所有服务${RESET}"
  2517. echo -e "2) ${BOLD}删除${LIGHT_CYAN}指定服务${RESET}"
  2518. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  2519. echo -e "0) ${BOLD}退出脚本${RESET}"
  2520. echo "---------------------------------------------------------------"
  2521. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" rm_choice
  2522. case $rm_choice in
  2523. 1)
  2524. CONFIREM_ACTION "卸载所有" RM_ALLSERVICE
  2525. ;;
  2526. 2)
  2527. CONFIREM_ACTION "删除指定" RM_SERVICE
  2528. UNI_DOCKER_SERVICE
  2529. ;;
  2530. 3)
  2531. main_menu
  2532. ;;
  2533. 0)
  2534. exit 1
  2535. ;;
  2536. *)
  2537. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  2538. UNI_DOCKER_SERVICE
  2539. ;;
  2540. esac
  2541. }
  2542. function AUTH_SERVICE_CONFIG() {
  2543. CHECK_COMPOSE_CMD
  2544. CHECK_REG_AUTH() {
  2545. if ! command -v docker &> /dev/null; then
  2546. ERROR "docker 命令未找到,请确保 Docker 已正确安装"
  2547. AUTH_SERVICE_CONFIG
  2548. fi
  2549. # 重置数组
  2550. running_containers=()
  2551. stopped_containers=()
  2552. auth_containers=()
  2553. non_auth_containers=()
  2554. declare -A services
  2555. services=(
  2556. ["reg-docker-hub"]="dockerhub"
  2557. ["reg-gcr"]="gcr"
  2558. ["reg-ghcr"]="ghcr"
  2559. ["reg-quay"]="quay"
  2560. ["reg-k8s-gcr"]="k8sgcr"
  2561. ["reg-k8s"]="k8s"
  2562. ["reg-mcr"]="mcr"
  2563. ["reg-elastic"]="elastic"
  2564. ["reg-nvcr"]="nvcr"
  2565. )
  2566. container_names=$(docker ps --filter "name=reg-" --filter "status=running" --format "{{.Names}}")
  2567. # 检查哪些容器正在运行
  2568. for container_name in "${!services[@]}"; do
  2569. if docker ps --filter "name=$container_name" --filter "status=running" --format "{{.Names}}" | grep -q "$container_name"; then
  2570. running_containers+=("${services[$container_name]}")
  2571. else
  2572. stopped_containers+=("${services[$container_name]}")
  2573. fi
  2574. done
  2575. if [ ${#running_containers[@]} -gt 0 ]; then
  2576. INFO "当前运行的 Docker 容器有: ${LIGHT_CYAN}${running_containers[*]}${RESET}"
  2577. else
  2578. WARN "当前没有运行的 Docker 容器"
  2579. AUTH_SERVICE_CONFIG # 如果没有运行的容器,直接返回
  2580. fi
  2581. auth_containers=()
  2582. non_auth_containers=()
  2583. for container_name in $container_names; do
  2584. specified_name=${services[$container_name]}
  2585. if [ -z "$specified_name" ]; then
  2586. specified_name=$container_name
  2587. fi
  2588. if docker exec $container_name grep -q "auth" /etc/distribution/config.yml; then
  2589. auth_containers+=("$specified_name")
  2590. else
  2591. non_auth_containers+=("$specified_name")
  2592. fi
  2593. done
  2594. if [ ${#auth_containers[@]} -gt 0 ]; then
  2595. INFO "当前运行的 Docker 容器中${LIGHT_GREEN}包含认证${RESET}的容器有: ${LIGHT_CYAN}${auth_containers[*]}${RESET}"
  2596. else
  2597. WARN "当前运行的 Docker 容器中${LIGHT_YELLOW}没有包含认证${RESET}的容器"
  2598. fi
  2599. if [ ${#non_auth_containers[@]} -gt 0 ]; then
  2600. WARN "当前运行的 Docker 容器中${LIGHT_YELLOW}没有包含认证${RESET}的容器有: ${LIGHT_BLUE}${non_auth_containers[*]}${RESET}"
  2601. else
  2602. INFO "当前运行的 Docker 容器中${LIGHT_GREEN}所有容器都包含认证${RESET}"
  2603. fi
  2604. }
  2605. AUTH_MENU() {
  2606. CHECK_REG_AUTH
  2607. selected_files=()
  2608. selected_services=()
  2609. REGISTRY_FILES
  2610. REGISTRY_SER_MENU
  2611. read -e -p "$(INFO "输入序号选择添加认证的服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" auth_service
  2612. while [[ ! "$auth_service" =~ ^([0-9]+[[:space:]]*)+$ ]]; do
  2613. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2614. read -e -p "$(INFO "输入序号选择添加认证的服务,${LIGHT_YELLOW}空格分隔${RESET}多个选项 > ")" auth_service
  2615. done
  2616. }
  2617. ADD_AUTH_CONFIG() {
  2618. local FILE=$1
  2619. local auth_config="
  2620. auth:
  2621. htpasswd:
  2622. realm: basic-realm
  2623. path: /auth/htpasswd"
  2624. if [ ! -f "$FILE" ]; then
  2625. ERROR "配置文件 ${LIGHT_BLUE}$FILE${RESET} 不存在"
  2626. exit 1
  2627. else
  2628. if ! grep -q "auth:" "$FILE" || ! grep -q "htpasswd:" "$FILE" || ! grep -q "realm: basic-realm" "$FILE" || ! grep -q "path: /auth/htpasswd" "$FILE"; then
  2629. echo -e "$auth_config" | sudo tee -a "$FILE" > /dev/null
  2630. INFO "配置文件 ${LIGHT_BLUE}$FILE${RESET} 添加认证配置成功"
  2631. else
  2632. WARN "配置文件 ${LIGHT_BLUE}$FILE${RESET} 已添加认证配置"
  2633. fi
  2634. fi
  2635. }
  2636. ADD_AUTH_COMPOSE() {
  2637. local SERVICES=$1
  2638. local FILE=${DOCKER_COMPOSE_FILE}
  2639. local HTPASSWD_CONFIG=" - ./${SERVICES}_htpasswd:/auth/htpasswd"
  2640. if [ ! -f "$FILE" ]; then
  2641. ERROR "配置文件 ${LIGHT_BLUE}$FILE${RESET} 不存在"
  2642. exit 1
  2643. fi
  2644. for SERVICE in "${SERVICES[@]}"; do
  2645. if grep -q " $SERVICE:" "$FILE"; then
  2646. if ! grep -A10 " $SERVICE:" "$FILE" | grep -q " - ./${SERVICES}_htpasswd:/auth/htpasswd"; then
  2647. sed -i "/ $SERVICE:/,/volumes:/ {
  2648. /volumes:/a\\
  2649. $HTPASSWD_CONFIG
  2650. }" "$FILE"
  2651. INFO "Htpasswd配置添加到 ${LIGHT_GREEN}$SERVICE${RESET} 服务中"
  2652. else
  2653. WARN "Htpasswd配置已存在 ${LIGHT_YELLOW}$SERVICE${RESET} 服务中"
  2654. fi
  2655. else
  2656. ERROR "服务 $SERVICE 在 $FILE 中不存在"
  2657. fi
  2658. done
  2659. }
  2660. DEL_AUTH_CONFIG() {
  2661. local FILE=$1
  2662. if [ ! -f "$FILE" ]; then
  2663. ERROR "配置文件 $FILE 不存在"
  2664. else
  2665. if grep -q "auth:" "$FILE"; then
  2666. sed -i '/^auth:$/,/^[^[:space:]]/d' "$FILE" >/dev/null
  2667. INFO "配置文件 ${LIGHT_BLUE}$FILE${RESET} 成功移除认证信息"
  2668. else
  2669. WARN "配置文件 ${LIGHT_BLUE}$FILE${RESET} 不存在认证信息"
  2670. fi
  2671. fi
  2672. }
  2673. DEL_AUTH_COMPOSE() {
  2674. local SERVICES=$1
  2675. local FILE=${DOCKER_COMPOSE_FILE}
  2676. if [ ! -f "$FILE" ]; then
  2677. ERROR "$File 不存在"
  2678. exit 1
  2679. fi
  2680. for SERVICE in "${SERVICES[@]}"; do
  2681. if grep -q " $SERVICE:" "$FILE"; then
  2682. sed -i "/ $SERVICE:/,/^[^[:space:]]/ {/^[[:space:]]*- .\/${SERVICES}_htpasswd:\/auth\/htpasswd/d}" "$FILE"
  2683. else
  2684. ERROR "$FILE 中不存在服务 $SERVICE"
  2685. fi
  2686. done
  2687. }
  2688. ENABLE_AUTH() {
  2689. AUTH_MENU
  2690. if [[ "$auth_service" == "0" ]]; then
  2691. WARN "退出添加容器认证操作!"
  2692. return
  2693. else
  2694. for choice in ${auth_service}; do
  2695. if [[ $choice =~ ^[0-9]+$ ]] && ((choice > 0 && choice <= ${#files[@]})); then
  2696. file_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f2)
  2697. service_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f1)
  2698. selected_files+=("$file_name")
  2699. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2700. selected_services+=("$service_name")
  2701. else
  2702. WARN "服务 ${LIGHT_MAGENTA}${service_name} 未运行${RESET},无法添加认证授权"
  2703. fi
  2704. else
  2705. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2706. AUTH_MENU
  2707. return
  2708. fi
  2709. done
  2710. WARN "${LIGHT_GREEN}>>> 提示:${RESET} ${LIGHT_CYAN}配置认证后,执行镜像拉取需先通过 docker login登入后使用.访问UI需输入账号密码${RESET}"
  2711. read -e -p "$(INFO "是否需要配置镜像仓库访问账号和密码? ${PROMPT_YES_NO}")" enable_auth
  2712. while [[ "$enable_auth" != "y" && "$enable_auth" != "n" ]]; do
  2713. WARN "无效输入,请输入 ${LIGHT_GREEN}y${RESET} 或 ${LIGHT_YELLOW}n${RESET}"
  2714. read -e -p "$(INFO "是否需要配置镜像仓库访问账号和密码? ${PROMPT_YES_NO}")" enable_auth
  2715. done
  2716. if [[ "$enable_auth" == "y" ]]; then
  2717. while true; do
  2718. read -e -p "$(INFO "请输入账号名称: ")" username
  2719. if [[ -z "$username" ]]; then
  2720. ERROR "用户名不能为空。请重新输入"
  2721. else
  2722. break
  2723. fi
  2724. done
  2725. while true; do
  2726. read -e -p "$(INFO "请输入账号密码: ")" password
  2727. if [[ -z "$password" ]]; then
  2728. ERROR "密码不能为空。请重新输入"
  2729. else
  2730. break
  2731. fi
  2732. done
  2733. for file_url in "${selected_files[@]}"; do
  2734. yml_name=$(basename "$file_url")
  2735. ADD_AUTH_CONFIG "${PROXY_DIR}/${yml_name}"
  2736. done
  2737. for server in "${selected_services[@]}"; do
  2738. htpasswd -Bbc ${PROXY_DIR}/${server}_htpasswd "$username" "$password" 2>/dev/null
  2739. ADD_AUTH_COMPOSE "${server}"
  2740. done
  2741. fi
  2742. fi
  2743. }
  2744. DELETE_AUTH() {
  2745. AUTH_MENU
  2746. if [[ "$auth_service" == "0" ]]; then
  2747. WARN "退出移除容器认证操作!"
  2748. return
  2749. else
  2750. for choice in ${auth_service}; do
  2751. if [[ $choice =~ ^[0-9]+$ ]] && ((choice > 0 && choice <= ${#files[@]})); then
  2752. file_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f2)
  2753. service_name=$(echo "${files[$((choice - 1))]}" | cut -d' ' -f1)
  2754. selected_files+=("$file_name")
  2755. if $DOCKER_COMPOSE_CMD ps --services 2>/dev/null | grep -q "^${service_name}$"; then
  2756. selected_services+=("$service_name")
  2757. else
  2758. WARN "服务 ${LIGHT_MAGENTA}${service_name} 未运行${RESET},无法添加认证授权"
  2759. fi
  2760. else
  2761. WARN "无效输入,请重新输入${LIGHT_YELLOW} 0-9 ${RESET}序号"
  2762. AUTH_MENU
  2763. return
  2764. fi
  2765. done
  2766. for file_url in "${selected_files[@]}"; do
  2767. yml_name=$(basename "$file_url")
  2768. DEL_AUTH_CONFIG "${PROXY_DIR}/${yml_name}"
  2769. done
  2770. for server in "${selected_services[@]}"; do
  2771. DEL_AUTH_COMPOSE "${server}"
  2772. rm -f ${PROXY_DIR}/${server}_htpasswd
  2773. done
  2774. fi
  2775. }
  2776. SEPARATOR "认证授权"
  2777. echo -e "1) ${BOLD}${LIGHT_YELLOW}添加${RESET}认证"
  2778. echo -e "2) ${BOLD}${LIGHT_CYAN}删除${RESET}认证"
  2779. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  2780. echo -e "0) ${BOLD}退出脚本${RESET}"
  2781. echo "---------------------------------------------------------------"
  2782. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" auth_choice
  2783. case $auth_choice in
  2784. 1)
  2785. ENABLE_AUTH
  2786. if [ ${#selected_services[@]} -eq 0 ]; then
  2787. WARN "退出认证授权,请${LIGHT_CYAN}重新选择${RESET}服务认证的操作"
  2788. AUTH_SERVICE_CONFIG # 没有服务运行调用函数
  2789. else
  2790. $DOCKER_COMPOSE_CMD down ${selected_services[*]}
  2791. $DOCKER_COMPOSE_CMD up -d --force-recreate ${selected_services[*]}
  2792. fi
  2793. AUTH_SERVICE_CONFIG
  2794. ;;
  2795. 2)
  2796. DELETE_AUTH
  2797. if [ ${#selected_services[@]} -eq 0 ]; then
  2798. WARN "退出认证授权,请${LIGHT_CYAN}重新选择${RESET}服务认证的操作"
  2799. AUTH_SERVICE_CONFIG # 没有服务运行调用函数
  2800. else
  2801. $DOCKER_COMPOSE_CMD down ${selected_services[*]}
  2802. $DOCKER_COMPOSE_CMD up -d --force-recreate ${selected_services[*]}
  2803. fi
  2804. AUTH_SERVICE_CONFIG
  2805. ;;
  2806. 3)
  2807. main_menu
  2808. ;;
  2809. 0)
  2810. exit 1
  2811. ;;
  2812. *)
  2813. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  2814. AUTH_SERVICE_CONFIG
  2815. ;;
  2816. esac
  2817. }
  2818. # 本机Docker代理
  2819. function DOCKER_PROXY() {
  2820. SEPARATOR "Docker服务代理"
  2821. echo -e "1) ${BOLD}${GREEN}添加${RESET}本机Docker代理"
  2822. echo -e "2) ${BOLD}${YELLOW}移除${RESET}本机Docker代理"
  2823. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  2824. echo -e "0) ${BOLD}退出脚本${RESET}"
  2825. echo "---------------------------------------------------------------"
  2826. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" main_choice
  2827. case $main_choice in
  2828. 1)
  2829. DOCKER_PROXY_HTTP
  2830. ADD_DOCKERD_PROXY
  2831. DOCKER_PROXY
  2832. ;;
  2833. 2)
  2834. DEL_DOCKERD_PROXY
  2835. DOCKER_PROXY
  2836. ;;
  2837. 3)
  2838. main_menu
  2839. ;;
  2840. 0)
  2841. exit 1
  2842. ;;
  2843. *)
  2844. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  2845. sleep 2; DOCKER_PROXY
  2846. ;;
  2847. esac
  2848. }
  2849. # IP 黑白名单
  2850. function IP_BLACKWHITE_LIST() {
  2851. if ! command -v iptables &> /dev/null
  2852. then
  2853. WARN "iptables 未安装. 请安装后再运行此脚本."
  2854. exit 1
  2855. fi
  2856. IPTABLES=$(which iptables)
  2857. BLACKLIST_CHAIN="IP_BLACKLIST"
  2858. WHITELIST_CHAIN="IP_WHITELIST"
  2859. WHITELIST_FILE="/etc/firewall/whitelist.txt"
  2860. BLACKLIST_FILE="/etc/firewall/blacklist.txt"
  2861. # 确保文件存在
  2862. mkdir -p /etc/firewall
  2863. touch $WHITELIST_FILE $BLACKLIST_FILE
  2864. get_chain_name() {
  2865. local chain=$1
  2866. case $chain in
  2867. $BLACKLIST_CHAIN) echo "黑名单" ;;
  2868. $WHITELIST_CHAIN) echo "白名单" ;;
  2869. *) echo "未知名单" ;;
  2870. esac
  2871. }
  2872. create_chains() {
  2873. $IPTABLES -N $BLACKLIST_CHAIN 2>/dev/null
  2874. $IPTABLES -N $WHITELIST_CHAIN 2>/dev/null
  2875. }
  2876. check_ip() {
  2877. local ip=$1
  2878. local ipv4_regex='^([0-9]{1,3}\.){3}[0-9]{1,3}$'
  2879. local ipv6_regex='^([0-9a-fA-F]{1,4}:){7}[0-9a-fA-F]{1,4}$'
  2880. if [[ $ip =~ $ipv4_regex ]] || [[ $ip =~ $ipv6_regex ]]; then
  2881. return 0
  2882. else
  2883. return 1
  2884. fi
  2885. }
  2886. ip_exists_in_file() {
  2887. local ip=$1
  2888. local file=$2
  2889. grep -q "^$ip$" "$file"
  2890. return $?
  2891. }
  2892. add_ips_to_file() {
  2893. local ips=("$@")
  2894. local file="${ips[-1]}"
  2895. unset 'ips[-1]'
  2896. local chain_name=$(get_chain_name $([[ $file == $WHITELIST_FILE ]] && echo $WHITELIST_CHAIN || echo $BLACKLIST_CHAIN))
  2897. local added=()
  2898. local skipped=()
  2899. for ip in "${ips[@]}"; do
  2900. if ! ip_exists_in_file $ip $file; then
  2901. echo $ip >> "$file"
  2902. added+=("$ip")
  2903. else
  2904. skipped+=("$ip")
  2905. fi
  2906. done
  2907. if [ ${#added[@]} -gt 0 ]; then
  2908. INFO "${LIGHT_BLUE}${added[*]}${RESET} ${LIGHT_GREEN}已添加${RESET}到$chain_name"
  2909. fi
  2910. if [ ${#skipped[@]} -gt 0 ]; then
  2911. WARN "${LIGHT_BLUE}${skipped[*]}${RESET} ${LIGHT_YELLOW}已存在${RESET}于$chain_name,跳过添加"
  2912. fi
  2913. }
  2914. remove_ips_from_file() {
  2915. local ips=("$@")
  2916. local file="${ips[-1]}"
  2917. unset 'ips[-1]'
  2918. local chain_name=$(get_chain_name $([[ $file == $WHITELIST_FILE ]] && echo $WHITELIST_CHAIN || echo $BLACKLIST_CHAIN))
  2919. local removed=()
  2920. local not_found=()
  2921. for ip in "${ips[@]}"; do
  2922. if ip_exists_in_file $ip $file; then
  2923. sed -i "/^$ip$/d" "$file"
  2924. removed+=("$ip")
  2925. else
  2926. not_found+=("$ip")
  2927. fi
  2928. done
  2929. if [ ${#removed[@]} -gt 0 ]; then
  2930. INFO "${LIGHT_BLUE}${removed[*]}${RESET} ${LIGHT_RED}已从${RESET}$chain_name${LIGHT_RED}移除${RESET}"
  2931. fi
  2932. if [ ${#not_found[@]} -gt 0 ]; then
  2933. WARN "${LIGHT_BLUE}${not_found[*]}${RESET} ${LIGHT_YELLOW}不存在${RESET}于$chain_name,无需移除"
  2934. fi
  2935. }
  2936. list_ips_in_file() {
  2937. local file=$1
  2938. local chain_name=$(get_chain_name $([[ $file == $WHITELIST_FILE ]] && echo $WHITELIST_CHAIN || echo $BLACKLIST_CHAIN))
  2939. echo "---------------------------------------------------------------"
  2940. echo "当前${chain_name}中的IP列表:"
  2941. cat "$file"
  2942. }
  2943. apply_ip_list() {
  2944. local chain=$1
  2945. local file=$2
  2946. local action=$3
  2947. # 清空链
  2948. $IPTABLES -F $chain
  2949. # 使用 iptables-restore 批量应用规则
  2950. {
  2951. echo "*filter"
  2952. echo ":$chain - [0:0]"
  2953. while IFS= read -r ip; do
  2954. echo "-A $chain -s $ip -j $action"
  2955. done < "$file"
  2956. echo "COMMIT"
  2957. } | $IPTABLES-restore -n
  2958. }
  2959. ensure_default_deny_for_whitelist() {
  2960. if ! $IPTABLES -C $WHITELIST_CHAIN -j DROP &>/dev/null; then
  2961. $IPTABLES -A $WHITELIST_CHAIN -j DROP
  2962. fi
  2963. }
  2964. whitelist_is_empty() {
  2965. [ ! -s "$WHITELIST_FILE" ]
  2966. }
  2967. apply_whitelist() {
  2968. if whitelist_is_empty; then
  2969. WARN "白名单为空,不应用白名单规则以避免锁定系统。"
  2970. return 1
  2971. fi
  2972. if ! $IPTABLES -C INPUT -j $WHITELIST_CHAIN &>/dev/null; then
  2973. $IPTABLES -I INPUT 1 -j $WHITELIST_CHAIN
  2974. INFO "已将白名单规则应用到 INPUT 链"
  2975. else
  2976. INFO "白名单规则已经应用到 INPUT 链"
  2977. fi
  2978. apply_ip_list $WHITELIST_CHAIN $WHITELIST_FILE ACCEPT
  2979. ensure_default_deny_for_whitelist
  2980. return 0
  2981. }
  2982. switch_to_whitelist() {
  2983. $IPTABLES -D INPUT -j $BLACKLIST_CHAIN 2>/dev/null
  2984. INFO "${LIGHT_YELLOW}已切换到白名单模式,请注意:请在添加IP后手动应用规则${RESET}"
  2985. }
  2986. switch_to_blacklist() {
  2987. $IPTABLES -D INPUT -j $WHITELIST_CHAIN 2>/dev/null
  2988. $IPTABLES -I INPUT 1 -j $BLACKLIST_CHAIN
  2989. apply_ip_list $BLACKLIST_CHAIN $BLACKLIST_FILE DROP
  2990. INFO "${LIGHT_YELLOW}已切换到黑名单模式${RESET}"
  2991. }
  2992. handle_whitelist() {
  2993. create_chains
  2994. local whitelist_mode_active=false
  2995. local whitelist_rules_applied=false
  2996. if $IPTABLES -C INPUT -j $WHITELIST_CHAIN &>/dev/null; then
  2997. whitelist_mode_active=true
  2998. whitelist_rules_applied=true
  2999. elif $IPTABLES -C INPUT -j $BLACKLIST_CHAIN &>/dev/null; then
  3000. read -e -p "$(WARN "${LIGHT_YELLOW}当前使用黑名单模式${RESET},${LIGHT_CYAN}是否切换到白名单模式?(y/n)${RESET}: ")" switch
  3001. if [[ $switch == "y" ]]; then
  3002. switch_to_whitelist
  3003. whitelist_mode_active=true
  3004. whitelist_rules_applied=false
  3005. else
  3006. WARN "保持在黑名单模式,返回主菜单。"
  3007. return
  3008. fi
  3009. else
  3010. switch_to_whitelist
  3011. whitelist_mode_active=true
  3012. whitelist_rules_applied=false
  3013. fi
  3014. while true; do
  3015. echo "---------------------------------------------------------------"
  3016. echo -e "1) ${BOLD}添加IP到白名单${RESET}"
  3017. echo -e "2) ${BOLD}从白名单移除IP${RESET}"
  3018. echo -e "3) ${BOLD}查看当前白名单${RESET}"
  3019. echo -e "4) ${BOLD}应用白名单规则${RESET}"
  3020. echo -e "5) ${BOLD}返回上一级${RESET}"
  3021. echo "---------------------------------------------------------------"
  3022. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" whitelist_choice
  3023. case $whitelist_choice in
  3024. 1)
  3025. read -e -p "$(INFO "${LIGHT_CYAN}请输入要添加到白名单的IP (用逗号分隔多个IP)${RESET}: ")" ips
  3026. IFS=',' read -ra ip_array <<< "$ips"
  3027. valid_ips=()
  3028. for ip in "${ip_array[@]}"; do
  3029. if check_ip $ip; then
  3030. valid_ips+=("$ip")
  3031. else
  3032. WARN "无效IP: $ip"
  3033. fi
  3034. done
  3035. if [ ${#valid_ips[@]} -gt 0 ]; then
  3036. add_ips_to_file "${valid_ips[@]}" "$WHITELIST_FILE"
  3037. whitelist_rules_applied=false
  3038. fi
  3039. ;;
  3040. 2)
  3041. read -e -p "$(INFO "${LIGHT_CYAN}请输入要从白名单移除的IP (用逗号分隔多个IP)${RESET}: ")" ips
  3042. IFS=',' read -ra ip_array <<< "$ips"
  3043. valid_ips=()
  3044. for ip in "${ip_array[@]}"; do
  3045. if check_ip $ip; then
  3046. valid_ips+=("$ip")
  3047. else
  3048. WARN "无效IP: $ip"
  3049. fi
  3050. done
  3051. if [ ${#valid_ips[@]} -gt 0 ]; then
  3052. remove_ips_from_file "${valid_ips[@]}" "$WHITELIST_FILE"
  3053. whitelist_rules_applied=false
  3054. fi
  3055. ;;
  3056. 3)
  3057. list_ips_in_file $WHITELIST_FILE
  3058. ;;
  3059. 4)
  3060. if apply_whitelist; then
  3061. whitelist_rules_applied=true
  3062. INFO "${LIGHT_GREEN}白名单规则已成功应用${RESET}"
  3063. else
  3064. WARN "${LIGHT_YELLOW}无法应用白名单规则${RESET}"
  3065. fi
  3066. ;;
  3067. 5)
  3068. if ! $whitelist_rules_applied; then
  3069. read -e -p "$(WARN "${LIGHT_YELLOW}白名单规则尚未应用。您确定要退出吗?${RESET} (y/n): ")" confirm
  3070. if [[ $confirm != "y" ]]; then
  3071. continue
  3072. fi
  3073. fi
  3074. return
  3075. ;;
  3076. *)
  3077. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择1-5${RESET}的选项."
  3078. ;;
  3079. esac
  3080. done
  3081. }
  3082. handle_blacklist() {
  3083. create_chains
  3084. local blacklist_mode_active=false
  3085. if $IPTABLES -C INPUT -j $BLACKLIST_CHAIN &>/dev/null; then
  3086. blacklist_mode_active=true
  3087. elif $IPTABLES -C INPUT -j $WHITELIST_CHAIN &>/dev/null; then
  3088. read -e -p "$(WARN "${LIGHT_YELLOW}当前使用白名单模式${RESET},${LIGHT_CYAN}是否切换到黑名单模式?(y/n)${RESET}: ")" switch
  3089. if [[ $switch == "y" ]]; then
  3090. switch_to_blacklist
  3091. blacklist_mode_active=true
  3092. else
  3093. WARN "保持在白名单模式,返回主菜单。"
  3094. return
  3095. fi
  3096. else
  3097. switch_to_blacklist
  3098. blacklist_mode_active=true
  3099. fi
  3100. while true; do
  3101. echo "---------------------------------------------------------------"
  3102. echo -e "1) ${BOLD}添加IP到黑名单${RESET}"
  3103. echo -e "2) ${BOLD}从黑名单移除IP${RESET}"
  3104. echo -e "3) ${BOLD}查看当前黑名单${RESET}"
  3105. echo -e "4) ${BOLD}返回上一级${RESET}"
  3106. echo "---------------------------------------------------------------"
  3107. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" blacklist_choice
  3108. case $blacklist_choice in
  3109. 1)
  3110. read -e -p "$(INFO "${LIGHT_CYAN}请输入要添加到黑名单的IP (用逗号分隔多个IP)${RESET}: ")" ips
  3111. IFS=',' read -ra ip_array <<< "$ips"
  3112. valid_ips=()
  3113. for ip in "${ip_array[@]}"; do
  3114. if check_ip $ip; then
  3115. valid_ips+=("$ip")
  3116. else
  3117. WARN "无效IP: $ip"
  3118. fi
  3119. done
  3120. if [ ${#valid_ips[@]} -gt 0 ]; then
  3121. add_ips_to_file "${valid_ips[@]}" "$BLACKLIST_FILE"
  3122. apply_ip_list $BLACKLIST_CHAIN $BLACKLIST_FILE DROP
  3123. fi
  3124. ;;
  3125. 2)
  3126. read -e -p "$(INFO "${LIGHT_CYAN}请输入要从黑名单移除的IP (用逗号分隔多个IP)${RESET}: ")" ips
  3127. IFS=',' read -ra ip_array <<< "$ips"
  3128. valid_ips=()
  3129. for ip in "${ip_array[@]}"; do
  3130. if check_ip $ip; then
  3131. valid_ips+=("$ip")
  3132. else
  3133. WARN "无效IP: $ip"
  3134. fi
  3135. done
  3136. if [ ${#valid_ips[@]} -gt 0 ]; then
  3137. remove_ips_from_file "${valid_ips[@]}" "$BLACKLIST_FILE"
  3138. apply_ip_list $BLACKLIST_CHAIN $BLACKLIST_FILE DROP
  3139. fi
  3140. ;;
  3141. 3)
  3142. list_ips_in_file $BLACKLIST_FILE
  3143. ;;
  3144. 4)
  3145. return
  3146. ;;
  3147. *)
  3148. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择1-4${RESET}的选项."
  3149. ;;
  3150. esac
  3151. done
  3152. }
  3153. while true; do
  3154. SEPARATOR "设置IP黑白名单"
  3155. echo -e "1) ${BOLD}管理${LIGHT_GREEN}白名单${RESET}"
  3156. echo -e "2) ${BOLD}管理${LIGHT_CYAN}黑名单${RESET}"
  3157. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  3158. echo -e "0) ${BOLD}退出脚本${RESET}"
  3159. echo "---------------------------------------------------------------"
  3160. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" ipblack_choice
  3161. case $ipblack_choice in
  3162. 1)
  3163. handle_whitelist
  3164. ;;
  3165. 2)
  3166. handle_blacklist
  3167. ;;
  3168. 3)
  3169. main_menu
  3170. ;;
  3171. 0)
  3172. exit 0
  3173. ;;
  3174. *)
  3175. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  3176. ;;
  3177. esac
  3178. done
  3179. }
  3180. # 其他工具
  3181. function OtherTools() {
  3182. SEPARATOR "其他工具"
  3183. echo -e "1) 设置${BOLD}${YELLOW}系统命令${RESET}"
  3184. echo -e "2) 配置${BOLD}${LIGHT_MAGENTA}IP黑白名单${RESET}"
  3185. echo -e "3) ${BOLD}返回${LIGHT_RED}主菜单${RESET}"
  3186. echo -e "0) ${BOLD}退出脚本${RESET}"
  3187. echo "---------------------------------------------------------------"
  3188. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" main_choice
  3189. case $main_choice in
  3190. 1)
  3191. ADD_SYS_CMD
  3192. ;;
  3193. 2)
  3194. IP_BLACKWHITE_LIST
  3195. ;;
  3196. 3)
  3197. main_menu
  3198. ;;
  3199. 0)
  3200. exit 1
  3201. ;;
  3202. *)
  3203. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-3${RESET}的选项."
  3204. sleep 2; main_menu
  3205. ;;
  3206. esac
  3207. }
  3208. ## 主菜单
  3209. function main_menu() {
  3210. echo -e "╔════════════════════════════════════════════════════╗"
  3211. echo -e "║ ║"
  3212. echo -e "║ ${LIGHT_CYAN}欢迎使用Docker-Proxy${RESET} ║"
  3213. echo -e "║ ║"
  3214. echo -e "║ TG频道: ${UNDERLINE}https://t.me/dqzboyblog${RESET} ║"
  3215. echo -e "║ ║"
  3216. echo -e "║ ${LIGHT_BLUE}by dqzboy${RESET} ║"
  3217. echo -e "║ ║"
  3218. echo -e "╚════════════════════════════════════════════════════╝"
  3219. echo
  3220. SEPARATOR "请选择操作"
  3221. echo -e "1) ${BOLD}${LIGHT_GREEN}安装${RESET}服务"
  3222. echo -e "2) ${BOLD}${LIGHT_MAGENTA}组件${RESET}安装"
  3223. echo -e "3) ${BOLD}${LIGHT_YELLOW}管理${RESET}服务"
  3224. echo -e "4) ${BOLD}${LIGHT_CYAN}更新${RESET}配置"
  3225. echo -e "5) ${BOLD}${LIGHT_RED}卸载${RESET}服务"
  3226. echo -e "6) ${BOLD}${LIGHT_BLUE}认证${RESET}授权"
  3227. echo -e "7) 本机${BOLD}${CYAN}Docker代理${RESET}"
  3228. echo -e "8) 其他${BOLD}${YELLOW}工具${RESET}"
  3229. echo -e "0) ${BOLD}退出脚本${RESET}"
  3230. echo "---------------------------------------------------------------"
  3231. read -e -p "$(INFO "输入${LIGHT_CYAN}对应数字${RESET}并按${LIGHT_GREEN}Enter${RESET}键 > ")" main_choice
  3232. case $main_choice in
  3233. 1)
  3234. INSTALL_PROXY
  3235. ;;
  3236. 2)
  3237. COMP_INST
  3238. ;;
  3239. 3)
  3240. SVC_MGMT
  3241. ;;
  3242. 4)
  3243. SEPARATOR "更新配置"
  3244. UPDATE_CONFIG
  3245. SEPARATOR "更新完成"
  3246. ;;
  3247. 5)
  3248. UNI_DOCKER_SERVICE
  3249. ;;
  3250. 6)
  3251. AUTH_SERVICE_CONFIG
  3252. ;;
  3253. 7)
  3254. DOCKER_PROXY
  3255. ;;
  3256. 8)
  3257. OtherTools
  3258. ;;
  3259. 0)
  3260. exit 1
  3261. ;;
  3262. *)
  3263. WARN "输入了无效的选择。请重新${LIGHT_GREEN}选择0-8${RESET}的选项."
  3264. sleep 2; main_menu
  3265. ;;
  3266. esac
  3267. }
  3268. main_menu