router_local_update.sh 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. #!/bin/sh
  2. set -eu
  3. timestamp() {
  4. date '+%Y-%m-%d %H:%M:%S'
  5. }
  6. log() {
  7. printf '[%s] %s\n' "$(timestamp)" "$*"
  8. }
  9. log_err() {
  10. printf '[%s] %s\n' "$(timestamp)" "$*" >&2
  11. }
  12. SCRIPT_DIR=$(CDPATH= cd -- "$(dirname -- "$0")" && pwd)
  13. APP_DIR=$(CDPATH= cd -- "$SCRIPT_DIR/.." && pwd)
  14. CONFIG_PATH=${1:-"$APP_DIR/config_router.conf"}
  15. if [ ! -r "$CONFIG_PATH" ]; then
  16. log_err "[router-local] config not found: $CONFIG_PATH"
  17. exit 1
  18. fi
  19. # shellcheck disable=SC1090
  20. . "$CONFIG_PATH"
  21. CFST_WORK_DIR=${CFST_WORK_DIR:-"$APP_DIR/cfst"}
  22. CFST_BIN=${CFST_BIN:-"./cfst"}
  23. CFST_IP_FILE=${CFST_IP_FILE:-"ip.txt"}
  24. CFST_RESULT_FILE=${CFST_RESULT_FILE:-"result.csv"}
  25. CFST_DISPLAY_COUNT=${CFST_DISPLAY_COUNT:-10}
  26. CFST_THREADS=${CFST_THREADS:-}
  27. CFST_TEST_COUNT=${CFST_TEST_COUNT:-}
  28. CFST_DOWNLOAD_COUNT=${CFST_DOWNLOAD_COUNT:-}
  29. CFST_DOWNLOAD_TIME=${CFST_DOWNLOAD_TIME:-}
  30. CFST_PORT=${CFST_PORT:-443}
  31. CFST_URL=${CFST_URL:-}
  32. CFST_HTTPING=${CFST_HTTPING:-0}
  33. CFST_HTTPING_CODE=${CFST_HTTPING_CODE:-}
  34. CFST_CFCOLO=${CFST_CFCOLO:-}
  35. CFST_LATENCY_LIMIT=${CFST_LATENCY_LIMIT:-}
  36. CFST_LATENCY_LOWER=${CFST_LATENCY_LOWER:-}
  37. CFST_LOSS_LIMIT=${CFST_LOSS_LIMIT:-}
  38. CFST_SPEED_LIMIT=${CFST_SPEED_LIMIT:-}
  39. CFST_DISABLE_DOWNLOAD=${CFST_DISABLE_DOWNLOAD:-0}
  40. CFST_ALL_IP=${CFST_ALL_IP:-0}
  41. CFST_DEBUG=${CFST_DEBUG:-0}
  42. CFST_SKIP_RUN=${CFST_SKIP_RUN:-0}
  43. CFST_QUIET_LOG=${CFST_QUIET_LOG:-1}
  44. CFST_SHOW_LOG_ON_TTY=${CFST_SHOW_LOG_ON_TTY:-1}
  45. TOP_N=${TOP_N:-3}
  46. RUNTIME_DIR=${RUNTIME_DIR:-"$APP_DIR/cfip_runtime"}
  47. VALUE_TEXT_FILE=${VALUE_TEXT_FILE:-"current_ip.txt"}
  48. VALUE_JSON_FILE=${VALUE_JSON_FILE:-"current_ip.json"}
  49. STATE_FILE=${STATE_FILE:-"state.json"}
  50. EXPORT_VARS_FILE=${EXPORT_VARS_FILE:-"substore_vars.json"}
  51. VALUE_JSON_KEY=${VALUE_JSON_KEY:-ip}
  52. STATE_LAST_GOOD_KEY=${STATE_LAST_GOOD_KEY:-last_good_ip}
  53. EXPORT_VALUE_KEY=${EXPORT_VALUE_KEY:-AUTO_CFIP}
  54. resolve_path() {
  55. _p="$1"
  56. _base_dir="$2"
  57. case "$_p" in
  58. /*) printf '%s\n' "$_p" ;;
  59. *) printf '%s/%s\n' "$_base_dir" "$_p" ;;
  60. esac
  61. }
  62. CURRENT_TEXT_PATH=$(resolve_path "$VALUE_TEXT_FILE" "$RUNTIME_DIR")
  63. CURRENT_JSON_PATH=$(resolve_path "$VALUE_JSON_FILE" "$RUNTIME_DIR")
  64. STATE_PATH=$(resolve_path "$STATE_FILE" "$RUNTIME_DIR")
  65. EXPORT_VARS_PATH=$(resolve_path "$EXPORT_VARS_FILE" "$RUNTIME_DIR")
  66. INDEX_PATH="$RUNTIME_DIR/index.html"
  67. RESULT_PATH=$(resolve_path "$CFST_RESULT_FILE" "$CFST_WORK_DIR")
  68. UPDATED_AT=$(date -u +"%Y-%m-%dT%H:%M:%SZ")
  69. mkdir -p "$RUNTIME_DIR"
  70. json_get_string() {
  71. key="$1"
  72. file="$2"
  73. if [ ! -f "$file" ]; then
  74. return 0
  75. fi
  76. sed -n "s/.*\"$key\":[[:space:]]*\"\([^\"]*\)\".*/\1/p" "$file" | head -n 1
  77. }
  78. json_escape() {
  79. printf '%s' "$1" | sed 's/\\/\\\\/g; s/"/\\"/g; s/ /\\t/g' | tr -d '\n\r'
  80. }
  81. build_top_candidates_json() {
  82. awk -F',' -v top_n="$TOP_N" '
  83. function trim(v) {
  84. gsub(/\r/, "", v)
  85. sub(/^[[:space:]]+/, "", v)
  86. sub(/[[:space:]]+$/, "", v)
  87. return v
  88. }
  89. function esc(v) {
  90. gsub(/\\/,"\\\\",v)
  91. gsub(/"/,"\\\"",v)
  92. return v
  93. }
  94. NR == 1 { next }
  95. count >= top_n { exit }
  96. NF < 1 { next }
  97. { if (trim($1) == "") next }
  98. {
  99. ip = esc(trim($1))
  100. sent = esc(trim($2))
  101. received = esc(trim($3))
  102. loss = esc(trim($4))
  103. latency = esc(trim($5))
  104. speed = esc(trim($6))
  105. region = esc(trim($7))
  106. if (count > 0) {
  107. printf(",")
  108. }
  109. printf("{\"domain\":\"%s\",\"ip\":\"%s\",\"source_type\":\"cfst_local_busybox\",\"sent\":\"%s\",\"received\":\"%s\",\"loss_rate\":\"%s\",\"avg_latency\":\"%s\",\"download_speed\":\"%s\",\"region\":\"%s\",\"location_country\":\"\",\"location_city\":\"\",\"host_provider\":\"\",\"score_value\":null,\"scores\":[],\"created_raw\":\"\"}", ip, ip, sent, received, loss, latency, speed, region)
  110. count++
  111. }
  112. END {
  113. if (count == 0) {
  114. printf("")
  115. }
  116. }
  117. ' "$RESULT_PATH"
  118. }
  119. write_text_value() {
  120. selected_ip="$1"
  121. printf '%s\n' "$selected_ip" > "$CURRENT_TEXT_PATH"
  122. rm -f "$INDEX_PATH"
  123. cp "$CURRENT_TEXT_PATH" "$INDEX_PATH"
  124. }
  125. write_success_outputs() {
  126. selected_ip="$1"
  127. top_candidates_json="$2"
  128. source_count="$3"
  129. escaped_ip=$(json_escape "$selected_ip")
  130. write_text_value "$selected_ip"
  131. cat > "$CURRENT_JSON_PATH" <<EOF
  132. {
  133. "$VALUE_JSON_KEY": "$escaped_ip",
  134. "updated_at": "$UPDATED_AT",
  135. "status": "ok",
  136. "source_type": "cfst_local_busybox",
  137. "source_count": $source_count,
  138. "checked_count": 0,
  139. "top_candidates": [${top_candidates_json}]
  140. }
  141. EOF
  142. if [ "$selected_ip" != "${LAST_GOOD_IP:-}" ] || [ ! -f "$STATE_PATH" ]; then
  143. cat > "$STATE_PATH" <<EOF
  144. {
  145. "updated_at": "$UPDATED_AT",
  146. "$STATE_LAST_GOOD_KEY": "$escaped_ip",
  147. "status": "ok",
  148. "source_count": $source_count,
  149. "checked_count": 0,
  150. "source_type": "cfst_local_busybox"
  151. }
  152. EOF
  153. fi
  154. cat > "$EXPORT_VARS_PATH" <<EOF
  155. {
  156. "$EXPORT_VALUE_KEY": "$escaped_ip",
  157. "UPDATED_AT": "$UPDATED_AT",
  158. "STATUS": "ok"
  159. }
  160. EOF
  161. }
  162. write_error_with_fallback() {
  163. last_good_ip="$1"
  164. error_message="$2"
  165. escaped_ip=$(json_escape "$last_good_ip")
  166. escaped_error=$(json_escape "$error_message")
  167. write_text_value "$last_good_ip"
  168. cat > "$CURRENT_JSON_PATH" <<EOF
  169. {
  170. "$VALUE_JSON_KEY": "$escaped_ip",
  171. "updated_at": "$UPDATED_AT",
  172. "status": "error_use_last_good",
  173. "error": "$escaped_error",
  174. "source_type": "cfst_local_busybox"
  175. }
  176. EOF
  177. if [ "$last_good_ip" != "${LAST_GOOD_IP:-}" ] || [ ! -f "$STATE_PATH" ]; then
  178. cat > "$STATE_PATH" <<EOF
  179. {
  180. "updated_at": "$UPDATED_AT",
  181. "$STATE_LAST_GOOD_KEY": "$escaped_ip",
  182. "status": "error",
  183. "error": "$escaped_error",
  184. "source_type": "cfst_local_busybox"
  185. }
  186. EOF
  187. fi
  188. cat > "$EXPORT_VARS_PATH" <<EOF
  189. {
  190. "$EXPORT_VALUE_KEY": "$escaped_ip",
  191. "UPDATED_AT": "$UPDATED_AT",
  192. "STATUS": "error_use_last_good"
  193. }
  194. EOF
  195. }
  196. run_cfst() (
  197. cd "$CFST_WORK_DIR"
  198. set -- "$CFST_BIN" -f "$CFST_IP_FILE" -o "$CFST_RESULT_FILE" -p "$CFST_DISPLAY_COUNT" -tp "$CFST_PORT"
  199. if [ -n "$CFST_THREADS" ]; then
  200. set -- "$@" -n "$CFST_THREADS"
  201. fi
  202. if [ -n "$CFST_TEST_COUNT" ]; then
  203. set -- "$@" -t "$CFST_TEST_COUNT"
  204. fi
  205. if [ -n "$CFST_DOWNLOAD_COUNT" ]; then
  206. set -- "$@" -dn "$CFST_DOWNLOAD_COUNT"
  207. fi
  208. if [ -n "$CFST_DOWNLOAD_TIME" ]; then
  209. set -- "$@" -dt "$CFST_DOWNLOAD_TIME"
  210. fi
  211. if [ -n "$CFST_URL" ]; then
  212. set -- "$@" -url "$CFST_URL"
  213. fi
  214. if [ "$CFST_HTTPING" = "1" ]; then
  215. set -- "$@" -httping
  216. fi
  217. if [ -n "$CFST_HTTPING_CODE" ]; then
  218. set -- "$@" -httping-code "$CFST_HTTPING_CODE"
  219. fi
  220. if [ -n "$CFST_CFCOLO" ]; then
  221. set -- "$@" -cfcolo "$CFST_CFCOLO"
  222. fi
  223. if [ -n "$CFST_LATENCY_LIMIT" ]; then
  224. set -- "$@" -tl "$CFST_LATENCY_LIMIT"
  225. fi
  226. if [ -n "$CFST_LATENCY_LOWER" ]; then
  227. set -- "$@" -tll "$CFST_LATENCY_LOWER"
  228. fi
  229. if [ -n "$CFST_LOSS_LIMIT" ]; then
  230. set -- "$@" -tlr "$CFST_LOSS_LIMIT"
  231. fi
  232. if [ -n "$CFST_SPEED_LIMIT" ]; then
  233. set -- "$@" -sl "$CFST_SPEED_LIMIT"
  234. fi
  235. if [ "$CFST_DISABLE_DOWNLOAD" = "1" ]; then
  236. set -- "$@" -dd
  237. fi
  238. if [ "$CFST_ALL_IP" = "1" ]; then
  239. set -- "$@" -allip
  240. fi
  241. if [ "$CFST_DEBUG" = "1" ]; then
  242. set -- "$@" -debug
  243. fi
  244. # Show cfst live logs for manual interactive runs, keep quiet for redirected/background runs.
  245. is_interactive_tty=0
  246. if [ "$CFST_SHOW_LOG_ON_TTY" = "1" ] && [ -t 1 ]; then
  247. is_interactive_tty=1
  248. fi
  249. if [ "$CFST_QUIET_LOG" = "1" ] && [ "$CFST_DEBUG" != "1" ] && [ "$is_interactive_tty" != "1" ]; then
  250. "$@" >/dev/null 2>&1
  251. else
  252. "$@"
  253. fi
  254. )
  255. log_update_summary() {
  256. selected_ip="$1"
  257. source_count="$2"
  258. log "[router-local] update_at=$UPDATED_AT selected_ip=$selected_ip candidates=$source_count"
  259. awk -F',' -v top_n="$TOP_N" '
  260. function trim(v) {
  261. gsub(/\r/, "", v)
  262. sub(/^[[:space:]]+/, "", v)
  263. sub(/[[:space:]]+$/, "", v)
  264. return v
  265. }
  266. NR == 1 { next }
  267. count >= top_n { exit }
  268. NF < 1 { next }
  269. { if (trim($1) == "") next }
  270. {
  271. ip = trim($1)
  272. sent = trim($2)
  273. received = trim($3)
  274. loss = trim($4)
  275. latency = trim($5)
  276. speed = trim($6)
  277. region = trim($7)
  278. count++
  279. printf("[router-local] candidate_%d ip=%s sent=%s recv=%s loss=%s latency=%s speed=%s region=%s\n", count, ip, sent, received, loss, latency, speed, region)
  280. }
  281. ' "$RESULT_PATH"
  282. }
  283. LAST_GOOD_IP=$(json_get_string "$STATE_LAST_GOOD_KEY" "$STATE_PATH")
  284. if [ "$CFST_SKIP_RUN" != "1" ]; then
  285. if ! run_cfst; then
  286. if [ -n "$LAST_GOOD_IP" ]; then
  287. write_error_with_fallback "$LAST_GOOD_IP" "cfst run failed"
  288. log "[router-local] cfst run failed, fallback to last good ip: $LAST_GOOD_IP"
  289. exit 0
  290. fi
  291. log_err "[router-local] cfst run failed and no last good ip available"
  292. exit 1
  293. fi
  294. fi
  295. if [ ! -s "$RESULT_PATH" ]; then
  296. if [ -n "$LAST_GOOD_IP" ]; then
  297. write_error_with_fallback "$LAST_GOOD_IP" "cfst result file missing or empty"
  298. log "[router-local] empty result, fallback to last good ip: $LAST_GOOD_IP"
  299. exit 0
  300. fi
  301. log_err "[router-local] result file missing or empty: $RESULT_PATH"
  302. exit 1
  303. fi
  304. BEST_LINE=$(sed -n '2p' "$RESULT_PATH" | tr -d '\r')
  305. BEST_IP=$(printf '%s' "$BEST_LINE" | cut -d',' -f1 | tr -d ' ')
  306. SOURCE_COUNT=$(awk -F',' 'NR > 1 && NF > 0 { count++ } END { print count + 0 }' "$RESULT_PATH")
  307. TOP_CANDIDATES_JSON=$(build_top_candidates_json)
  308. if [ -z "$BEST_IP" ]; then
  309. if [ -n "$LAST_GOOD_IP" ]; then
  310. write_error_with_fallback "$LAST_GOOD_IP" "no valid ip found in cfst result"
  311. log "[router-local] no valid ip found, fallback to last good ip: $LAST_GOOD_IP"
  312. exit 0
  313. fi
  314. log_err "[router-local] no valid ip found in result file"
  315. exit 1
  316. fi
  317. write_success_outputs "$BEST_IP" "$TOP_CANDIDATES_JSON" "$SOURCE_COUNT"
  318. log_update_summary "$BEST_IP" "$SOURCE_COUNT"