test_upload_api.sh 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365
  1. #!/bin/bash
  2. # ============================================
  3. # 文件上传接口测试脚本
  4. # ============================================
  5. # 使用方法: ./test_upload_api.sh [host] [port]
  6. # 示例: ./test_upload_api.sh localhost 5232
  7. # ============================================
  8. # 配置参数
  9. HOST=${1:-localhost}
  10. PORT=${2:-5232}
  11. BASE_URL="http://${HOST}:${PORT}"
  12. UPLOAD_URL="${BASE_URL}/api/v1/parse/upload"
  13. STATUS_URL="${BASE_URL}/parse/status"
  14. REGISTER_URL="${BASE_URL}/auth/register"
  15. # 测试文件路径(相对于脚本所在目录)
  16. SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
  17. TEST_FILE="${SCRIPT_DIR}/test.docx"
  18. # 测试用户信息
  19. TIMESTAMP=$(date +%s)
  20. TEST_USERNAME="testuser_${TIMESTAMP}"
  21. TEST_EMAIL="testuser_${TIMESTAMP}@test.com"
  22. TEST_PASSWORD="Test123456!"
  23. USER_ID=""
  24. # 颜色定义
  25. RED='\033[0;31m'
  26. GREEN='\033[0;32m'
  27. YELLOW='\033[1;33m'
  28. BLUE='\033[0;34m'
  29. NC='\033[0m' # No Color
  30. # 输出函数
  31. print_header() {
  32. echo -e "\n${BLUE}============================================${NC}"
  33. echo -e "${BLUE}$1${NC}"
  34. echo -e "${BLUE}============================================${NC}"
  35. }
  36. print_success() {
  37. echo -e "${GREEN}✓ $1${NC}"
  38. }
  39. print_error() {
  40. echo -e "${RED}✗ $1${NC}"
  41. }
  42. print_info() {
  43. echo -e "${YELLOW}➤ $1${NC}"
  44. }
  45. # 检查依赖
  46. check_dependencies() {
  47. print_header "检查依赖"
  48. if ! command -v curl &> /dev/null; then
  49. print_error "curl 未安装"
  50. exit 1
  51. fi
  52. print_success "curl 已安装"
  53. if ! command -v jq &> /dev/null; then
  54. print_info "jq 未安装,JSON格式化将不可用"
  55. JQ_AVAILABLE=false
  56. else
  57. print_success "jq 已安装"
  58. JQ_AVAILABLE=true
  59. fi
  60. }
  61. # 检查测试文件
  62. check_test_file() {
  63. print_header "检查测试文件"
  64. if [ ! -f "$TEST_FILE" ]; then
  65. print_error "测试文件不存在: $TEST_FILE"
  66. exit 1
  67. fi
  68. FILE_SIZE=$(stat -c%s "$TEST_FILE" 2>/dev/null || stat -f%z "$TEST_FILE" 2>/dev/null)
  69. print_success "测试文件存在: $TEST_FILE"
  70. print_info "文件大小: $FILE_SIZE bytes"
  71. }
  72. # 检查服务是否可用
  73. check_service() {
  74. print_header "检查服务状态"
  75. print_info "测试服务: $BASE_URL"
  76. HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" --connect-timeout 5 "${BASE_URL}/actuator/health" 2>/dev/null)
  77. if [ "$HTTP_CODE" = "200" ]; then
  78. print_success "服务正常运行 (HTTP $HTTP_CODE)"
  79. elif [ "$HTTP_CODE" = "000" ]; then
  80. print_error "无法连接到服务 $BASE_URL"
  81. print_info "请确保 parse-service 正在运行"
  82. exit 1
  83. else
  84. print_info "健康检查返回 HTTP $HTTP_CODE,继续测试..."
  85. fi
  86. }
  87. # 注册测试用户
  88. register_test_user() {
  89. print_header "注册测试用户"
  90. print_info "用户名: $TEST_USERNAME"
  91. print_info "邮箱: $TEST_EMAIL"
  92. print_info "注册URL: $REGISTER_URL"
  93. RESPONSE=$(curl -s -w "\n%{http_code}" \
  94. -X POST "$REGISTER_URL" \
  95. -H "Content-Type: application/json" \
  96. -d "{\"username\":\"${TEST_USERNAME}\",\"email\":\"${TEST_EMAIL}\",\"password\":\"${TEST_PASSWORD}\",\"confirmPassword\":\"${TEST_PASSWORD}\"}" \
  97. --connect-timeout 10)
  98. HTTP_CODE=$(echo "$RESPONSE" | tail -n1)
  99. BODY=$(echo "$RESPONSE" | sed '$d')
  100. echo -e "\n${YELLOW}响应状态码:${NC} $HTTP_CODE"
  101. if [ "$JQ_AVAILABLE" = true ]; then
  102. echo "$BODY" | jq . 2>/dev/null || echo "$BODY"
  103. else
  104. echo "$BODY"
  105. fi
  106. # 解析用户ID
  107. if [ "$HTTP_CODE" = "200" ] || [ "$HTTP_CODE" = "201" ]; then
  108. print_success "用户注册成功!"
  109. if [ "$JQ_AVAILABLE" = true ]; then
  110. USER_ID=$(echo "$BODY" | jq -r '.data.user.id // .data.userId // .userId // empty' 2>/dev/null)
  111. if [ -z "$USER_ID" ] || [ "$USER_ID" = "null" ]; then
  112. # 尝试其他可能的字段
  113. USER_ID=$(echo "$BODY" | jq -r '.data.id // .id // empty' 2>/dev/null)
  114. fi
  115. if [ -n "$USER_ID" ] && [ "$USER_ID" != "null" ]; then
  116. print_info "用户ID: $USER_ID"
  117. echo "$USER_ID" > "${SCRIPT_DIR}/.last_user_id"
  118. else
  119. print_error "无法从响应中获取用户ID"
  120. echo "响应内容: $BODY"
  121. exit 1
  122. fi
  123. fi
  124. else
  125. print_error "用户注册失败 (HTTP $HTTP_CODE)"
  126. print_info "响应: $BODY"
  127. exit 1
  128. fi
  129. }
  130. # 测试文件上传
  131. test_upload() {
  132. print_header "测试文件上传接口"
  133. print_info "上传URL: $UPLOAD_URL"
  134. print_info "用户ID: $USER_ID"
  135. print_info "文件: $TEST_FILE"
  136. echo -e "\n发送请求..."
  137. RESPONSE=$(curl -s -w "\n%{http_code}" \
  138. -X POST "$UPLOAD_URL" \
  139. -H "Content-Type: multipart/form-data" \
  140. -F "file=@${TEST_FILE}" \
  141. -F "userId=${USER_ID}" \
  142. --connect-timeout 10 \
  143. --max-time 300)
  144. HTTP_CODE=$(echo "$RESPONSE" | tail -n1)
  145. BODY=$(echo "$RESPONSE" | sed '$d')
  146. echo -e "\n${YELLOW}响应状态码:${NC} $HTTP_CODE"
  147. echo -e "${YELLOW}响应内容:${NC}"
  148. if [ "$JQ_AVAILABLE" = true ]; then
  149. echo "$BODY" | jq . 2>/dev/null || echo "$BODY"
  150. else
  151. echo "$BODY"
  152. fi
  153. # 解析响应
  154. if [ "$HTTP_CODE" = "200" ]; then
  155. print_success "文件上传成功!"
  156. # 提取documentId用于后续状态查询
  157. if [ "$JQ_AVAILABLE" = true ]; then
  158. DOCUMENT_ID=$(echo "$BODY" | jq -r '.data.documentId // .documentId // empty' 2>/dev/null)
  159. if [ -n "$DOCUMENT_ID" ] && [ "$DOCUMENT_ID" != "null" ]; then
  160. print_info "文档ID: $DOCUMENT_ID"
  161. echo "$DOCUMENT_ID" > "${SCRIPT_DIR}/.last_document_id"
  162. # 查询解析状态
  163. test_parse_status "$DOCUMENT_ID"
  164. fi
  165. fi
  166. else
  167. print_error "文件上传失败 (HTTP $HTTP_CODE)"
  168. fi
  169. }
  170. # 测试解析状态查询
  171. test_parse_status() {
  172. local DOC_ID=$1
  173. print_header "查询解析状态"
  174. print_info "文档ID: $DOC_ID"
  175. print_info "状态URL: ${STATUS_URL}/${DOC_ID}"
  176. # 等待一会儿让解析任务开始
  177. sleep 2
  178. RESPONSE=$(curl -s -w "\n%{http_code}" \
  179. -X GET "${STATUS_URL}/${DOC_ID}" \
  180. --connect-timeout 10)
  181. HTTP_CODE=$(echo "$RESPONSE" | tail -n1)
  182. BODY=$(echo "$RESPONSE" | sed '$d')
  183. echo -e "\n${YELLOW}响应状态码:${NC} $HTTP_CODE"
  184. echo -e "${YELLOW}响应内容:${NC}"
  185. if [ "$JQ_AVAILABLE" = true ]; then
  186. echo "$BODY" | jq . 2>/dev/null || echo "$BODY"
  187. else
  188. echo "$BODY"
  189. fi
  190. if [ "$HTTP_CODE" = "200" ]; then
  191. print_success "状态查询成功!"
  192. else
  193. print_error "状态查询失败 (HTTP $HTTP_CODE)"
  194. fi
  195. }
  196. # 轮询解析状态直到完成
  197. poll_parse_status() {
  198. local DOC_ID=$1
  199. local MAX_ATTEMPTS=${2:-30}
  200. local INTERVAL=${3:-5}
  201. print_header "轮询解析状态 (最多${MAX_ATTEMPTS}次, 间隔${INTERVAL}秒)"
  202. for ((i=1; i<=MAX_ATTEMPTS; i++)); do
  203. print_info "第 $i 次查询..."
  204. RESPONSE=$(curl -s "${STATUS_URL}/${DOC_ID}" --connect-timeout 10)
  205. if [ "$JQ_AVAILABLE" = true ]; then
  206. STATUS=$(echo "$RESPONSE" | jq -r '.data.parseStatus // .parseStatus // empty' 2>/dev/null)
  207. echo "当前状态: $STATUS"
  208. if [ "$STATUS" = "2" ] || [ "$STATUS" = "COMPLETED" ]; then
  209. print_success "解析完成!"
  210. echo "$RESPONSE" | jq .
  211. return 0
  212. elif [ "$STATUS" = "3" ] || [ "$STATUS" = "FAILED" ]; then
  213. print_error "解析失败!"
  214. echo "$RESPONSE" | jq .
  215. return 1
  216. fi
  217. else
  218. echo "$RESPONSE"
  219. fi
  220. sleep $INTERVAL
  221. done
  222. print_error "轮询超时"
  223. return 1
  224. }
  225. # 显示使用帮助
  226. show_help() {
  227. echo "使用方法: $0 [选项] [host] [port]"
  228. echo ""
  229. echo "选项:"
  230. echo " -h, --help 显示帮助信息"
  231. echo " -p, --poll 上传后轮询解析状态直到完成"
  232. echo " -s, --status 仅查询上次上传的文档状态"
  233. echo ""
  234. echo "示例:"
  235. echo " $0 # 使用默认配置 (localhost:5232)"
  236. echo " $0 192.168.1.100 5232 # 指定服务器地址"
  237. echo " $0 -p # 上传并轮询状态"
  238. echo " $0 -s # 查询上次上传的状态"
  239. }
  240. # 主函数
  241. main() {
  242. local POLL_STATUS=false
  243. local STATUS_ONLY=false
  244. # 解析参数
  245. while [[ $# -gt 0 ]]; do
  246. case $1 in
  247. -h|--help)
  248. show_help
  249. exit 0
  250. ;;
  251. -p|--poll)
  252. POLL_STATUS=true
  253. shift
  254. ;;
  255. -s|--status)
  256. STATUS_ONLY=true
  257. shift
  258. ;;
  259. *)
  260. if [[ -z "$HOST_ARG" ]]; then
  261. HOST=$1
  262. elif [[ -z "$PORT_ARG" ]]; then
  263. PORT=$1
  264. fi
  265. shift
  266. ;;
  267. esac
  268. done
  269. # 更新URL
  270. BASE_URL="http://${HOST}:${PORT}"
  271. UPLOAD_URL="${BASE_URL}/api/v1/parse/upload"
  272. STATUS_URL="${BASE_URL}/parse/status"
  273. REGISTER_URL="${BASE_URL}/auth/register"
  274. print_header "文件上传接口测试"
  275. echo "目标服务: $BASE_URL"
  276. echo "时间: $(date '+%Y-%m-%d %H:%M:%S')"
  277. check_dependencies
  278. if [ "$STATUS_ONLY" = true ]; then
  279. if [ -f "${SCRIPT_DIR}/.last_document_id" ]; then
  280. DOCUMENT_ID=$(cat "${SCRIPT_DIR}/.last_document_id")
  281. test_parse_status "$DOCUMENT_ID"
  282. else
  283. print_error "未找到上次上传的文档ID"
  284. exit 1
  285. fi
  286. exit 0
  287. fi
  288. check_test_file
  289. check_service
  290. register_test_user
  291. test_upload
  292. if [ "$POLL_STATUS" = true ] && [ -f "${SCRIPT_DIR}/.last_document_id" ]; then
  293. DOCUMENT_ID=$(cat "${SCRIPT_DIR}/.last_document_id")
  294. poll_parse_status "$DOCUMENT_ID"
  295. fi
  296. print_header "测试完成"
  297. }
  298. # 运行主函数
  299. main "$@"