luat_ftp_client.c 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102
  1. /*
  2. @module ftp
  3. @summary ftp 客户端
  4. @version 1.0
  5. @date 2022.09.05
  6. @demo ftp
  7. @tag LUAT_USE_FTP
  8. */
  9. #include "luat_base.h"
  10. #include "luat_network_adapter.h"
  11. #include "luat_rtos.h"
  12. #include "luat_msgbus.h"
  13. #include "luat_fs.h"
  14. #include "luat_malloc.h"
  15. #include "luat_ftp.h"
  16. #define LUAT_LOG_TAG "ftp"
  17. #include "luat_log.h"
  18. #define FTP_DEBUG 0
  19. #if FTP_DEBUG == 0
  20. #undef LLOGD
  21. #define LLOGD(...)
  22. #endif
  23. enum
  24. {
  25. FTP_REQ_LOGIN = 1,
  26. FTP_REQ_COMMAND ,
  27. FTP_REQ_PULL ,
  28. FTP_REQ_PUSH ,
  29. FTP_REQ_CLOSE ,
  30. FTP_EVENT_LOGIN = USER_EVENT_ID_START + FTP_REQ_LOGIN,
  31. FTP_EVENT_COMMAND ,
  32. FTP_EVENT_PULL ,
  33. FTP_EVENT_PUSH ,
  34. FTP_EVENT_CLOSE ,
  35. FTP_EVENT_DATA_CONNECT ,
  36. FTP_EVENT_DATA_TX_DONE ,
  37. FTP_EVENT_DATA_WRITE_FILE ,
  38. FTP_EVENT_DATA_CLOSED ,
  39. };
  40. typedef struct{
  41. network_ctrl_t *cmd_netc; // ftp netc
  42. network_ctrl_t *data_netc; // ftp data_netc
  43. luat_ip_addr_t ip_addr; // ftp ip
  44. char addr[64]; // ftp addr
  45. char username[64]; // ftp username
  46. char password[64]; // ftp password
  47. char remote_name[64];//去掉?
  48. size_t upload_done_size;
  49. size_t local_file_size;
  50. uint8_t cmd_send_data[FTP_CMD_SEND_MAX];
  51. uint32_t cmd_send_len;
  52. uint8_t cmd_recv_data[FTP_CMD_RECV_MAX];
  53. uint32_t cmd_recv_len;
  54. uint16_t port; // 端口号
  55. uint8_t is_tls; // 是否SSL
  56. uint8_t adapter_index;
  57. uint8_t data_netc_online;
  58. uint8_t data_netc_connecting;
  59. }luat_ftp_network_t;
  60. typedef struct{
  61. uint64_t idp;
  62. luat_rtos_task_handle task_handle;
  63. luat_ftp_network_t *network;
  64. FILE* fd; //下载 FILE
  65. Buffer_Struct result_buffer;
  66. uint8_t is_run;
  67. }luat_ftp_ctrl_t;
  68. static luat_ftp_ctrl_t g_s_ftp;
  69. static uint32_t luat_ftp_close(void) {
  70. if (!g_s_ftp.network) return 0;
  71. if (g_s_ftp.network->cmd_netc){
  72. if (network_close(g_s_ftp.network->cmd_netc,FTP_SOCKET_TIMEOUT) < 0)
  73. {
  74. network_force_close_socket(g_s_ftp.network->cmd_netc);
  75. }
  76. network_release_ctrl(g_s_ftp.network->cmd_netc);
  77. g_s_ftp.network->cmd_netc = NULL;
  78. }
  79. if (g_s_ftp.network->data_netc){
  80. network_force_close_socket(g_s_ftp.network->data_netc);
  81. network_release_ctrl(g_s_ftp.network->data_netc);
  82. g_s_ftp.network->data_netc = NULL;
  83. }
  84. luat_heap_free(g_s_ftp.network);
  85. g_s_ftp.network = NULL;
  86. return 0;
  87. }
  88. static uint32_t luat_ftp_data_send(luat_ftp_ctrl_t *ftp_ctrl, uint8_t* send_data, uint32_t send_len) {
  89. if (send_len == 0)
  90. return 0;
  91. uint32_t tx_len = 0;
  92. LLOGD("luat_ftp_data_send data:%d",send_len);
  93. network_tx(g_s_ftp.network->data_netc, send_data, send_len, 0, NULL, 0, &tx_len, 0);
  94. return tx_len;
  95. }
  96. static uint32_t luat_ftp_cmd_send(luat_ftp_ctrl_t *ftp_ctrl, uint8_t* send_data, uint32_t send_len,uint32_t timeout_ms) {
  97. if (send_len == 0)
  98. return 0;
  99. uint32_t tx_len = 0;
  100. LLOGD("luat_ftp_cmd_send data:%.*s",send_len,send_data);
  101. network_tx(g_s_ftp.network->cmd_netc, send_data, send_len, 0, NULL, 0, &tx_len, timeout_ms);
  102. return tx_len;
  103. }
  104. static int luat_ftp_cmd_recv(luat_ftp_ctrl_t *ftp_ctrl,uint8_t *recv_data,uint32_t *recv_len,uint32_t timeout_ms){
  105. uint32_t total_len = 0;
  106. uint8_t is_break = 0,is_timeout = 0;
  107. int ret = network_wait_rx(g_s_ftp.network->cmd_netc, timeout_ms, &is_break, &is_timeout);
  108. LLOGD("network_wait_rx ret:%d is_break:%d is_timeout:%d",ret,is_break,is_timeout);
  109. if (ret)
  110. return -1;
  111. if (is_timeout)
  112. return 1;
  113. else if (is_break)
  114. return 2;
  115. int result = network_rx(g_s_ftp.network->cmd_netc, NULL, 0, 0, NULL, NULL, &total_len);
  116. if (0 == result){
  117. if (total_len>0){
  118. next:
  119. result = network_rx(g_s_ftp.network->cmd_netc, recv_data, total_len, 0, NULL, NULL, recv_len);
  120. LLOGD("result:%d recv_len:%d",result,*recv_len);
  121. LLOGD("recv_data %.*s",total_len, recv_data);
  122. if (result)
  123. goto next;
  124. if (*recv_len == 0||result!=0) {
  125. return -1;
  126. }
  127. return 0;
  128. }
  129. }else{
  130. LLOGE("ftp network_rx fail");
  131. return -1;
  132. }
  133. return 0;
  134. }
  135. static int32_t l_ftp_callback(lua_State *L, void* ptr){
  136. rtos_msg_t* msg = (rtos_msg_t*)lua_topointer(L, -1);
  137. LLOGD("l_ftp_callback arg1:%d arg2:%d idp:%lld",msg->arg1,msg->arg2,g_s_ftp.idp);
  138. if (g_s_ftp.idp)
  139. {
  140. if (msg->arg1)
  141. {
  142. lua_pushboolean(L, 0);
  143. }
  144. else if (msg->arg2)
  145. {
  146. lua_pushlstring(L,(const char *)(g_s_ftp.result_buffer.Data),g_s_ftp.result_buffer.Pos);
  147. }
  148. else
  149. {
  150. lua_pushboolean(L, 1);
  151. }
  152. luat_cbcwait(L, g_s_ftp.idp, 1);
  153. g_s_ftp.idp = 0;
  154. }
  155. OS_DeInitBuffer(&g_s_ftp.result_buffer);
  156. if (g_s_ftp.fd){
  157. luat_fs_fclose(g_s_ftp.fd);
  158. g_s_ftp.fd = NULL;
  159. }
  160. return 0;
  161. }
  162. static int32_t luat_ftp_data_callback(void *data, void *param){
  163. OS_EVENT *event = (OS_EVENT *)data;
  164. uint8_t *rx_buffer;
  165. int ret = 0;
  166. uint32_t rx_len = 0;
  167. if (!g_s_ftp.network)
  168. {
  169. return 0;
  170. }
  171. // LLOGD("event->ID %d LINK %d ON_LINE %d EVENT %d TX_OK %d CLOSED %d",event->ID,EV_NW_RESULT_LINK & 0x0fffffff,EV_NW_RESULT_CONNECT & 0x0fffffff,EV_NW_RESULT_EVENT & 0x0fffffff,EV_NW_RESULT_TX & 0x0fffffff,EV_NW_RESULT_CLOSE & 0x0fffffff);
  172. LLOGD("luat_ftp_data_callback %d %d",event->ID - EV_NW_RESULT_BASE, event->Param1);
  173. if (event->Param1){
  174. if (EV_NW_RESULT_CONNECT == event->ID)
  175. {
  176. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_DATA_CONNECT, 0xffffffff, 0, 0, LUAT_WAIT_FOREVER);
  177. }
  178. else
  179. {
  180. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_DATA_CLOSED, 0, 0, 0, LUAT_WAIT_FOREVER);
  181. }
  182. return -1;
  183. }
  184. switch (event->ID)
  185. {
  186. case EV_NW_RESULT_TX:
  187. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_DATA_TX_DONE, 0, 0, 0, LUAT_WAIT_FOREVER);
  188. break;
  189. case EV_NW_RESULT_EVENT:
  190. rx_buffer = NULL;
  191. uint8_t tmpbuff[4];
  192. do
  193. {
  194. // 先读取长度
  195. ret = network_rx(g_s_ftp.network->data_netc, NULL, 0, 0, NULL, NULL, &rx_len);
  196. if (rx_len <= 0) {
  197. // 没数据? 那也读一次, 然后退出
  198. network_rx(g_s_ftp.network->data_netc, tmpbuff, 4, 0, NULL, NULL, &rx_len);
  199. break;
  200. }
  201. if (rx_len > 2048)
  202. rx_len = 2048;
  203. rx_buffer = luat_heap_malloc(rx_len);
  204. // 如果rx_buffer == NULL, 内存炸了
  205. if (rx_buffer == NULL) {
  206. LLOGE("out of memory when malloc ftp buff");
  207. network_close(g_s_ftp.network->data_netc, 0);
  208. return -1;
  209. }
  210. ret = network_rx(g_s_ftp.network->data_netc, rx_buffer, rx_len, 0, NULL, NULL, &rx_len);
  211. // LLOGD("luat_ftp_data_callback network_rx ret:%d rx_len:%d",ret,rx_len);
  212. if (!ret && rx_len > 0)
  213. {
  214. if (g_s_ftp.fd)
  215. {
  216. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_DATA_WRITE_FILE, rx_buffer, rx_len, 0, LUAT_WAIT_FOREVER);
  217. rx_buffer = NULL;
  218. continue;
  219. }
  220. else
  221. {
  222. OS_BufferWrite(&g_s_ftp.result_buffer, rx_buffer, rx_len);
  223. }
  224. }
  225. luat_heap_free(rx_buffer);
  226. rx_buffer = NULL;
  227. } while (!ret && rx_len);
  228. if (rx_buffer)
  229. luat_heap_free(rx_buffer);
  230. rx_buffer = NULL;
  231. break;
  232. case EV_NW_RESULT_CLOSE:
  233. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_DATA_CLOSED, 0, 0, 0, LUAT_WAIT_FOREVER);
  234. return 0;
  235. break;
  236. case EV_NW_RESULT_CONNECT:
  237. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_DATA_CONNECT, 0, 0, 0, LUAT_WAIT_FOREVER);
  238. break;
  239. case EV_NW_RESULT_LINK:
  240. return 0;
  241. }
  242. ret = network_wait_event(g_s_ftp.network->data_netc, NULL, 0, NULL);
  243. if (ret < 0){
  244. network_close(g_s_ftp.network->data_netc, 0);
  245. return -1;
  246. }
  247. return 0;
  248. }
  249. static int32_t ftp_task_cb(void *pdata, void *param)
  250. {
  251. OS_EVENT *event = pdata;
  252. if (event->ID >= FTP_EVENT_LOGIN && event->ID <= FTP_EVENT_PUSH)
  253. {
  254. LLOGE("last cmd not finish, ignore %d,%u,%u,%x", event->ID - USER_EVENT_ID_START, event->Param1, event->Param2, param);
  255. return -1;
  256. }
  257. switch(event->ID)
  258. {
  259. case FTP_EVENT_DATA_WRITE_FILE:
  260. if (g_s_ftp.fd)
  261. {
  262. luat_fs_fwrite((void*)event->Param1, event->Param2, 1, g_s_ftp.fd);
  263. luat_heap_free((void*)event->Param1);
  264. }
  265. break;
  266. case FTP_EVENT_DATA_TX_DONE:
  267. g_s_ftp.network->upload_done_size = (size_t)g_s_ftp.network->data_netc->ack_size;
  268. if (g_s_ftp.network->upload_done_size >= g_s_ftp.network->local_file_size)
  269. {
  270. LLOGD("ftp data upload done!");
  271. network_close(g_s_ftp.network->data_netc, 0);
  272. }
  273. break;
  274. case FTP_EVENT_DATA_CONNECT:
  275. if (g_s_ftp.network->data_netc_connecting)
  276. {
  277. g_s_ftp.network->data_netc_connecting = 0;
  278. g_s_ftp.network->data_netc_online = !event->Param1;
  279. }
  280. break;
  281. case FTP_EVENT_DATA_CLOSED:
  282. LLOGD("ftp data channel close");
  283. g_s_ftp.network->data_netc_online = 0;
  284. if (g_s_ftp.network->data_netc)
  285. {
  286. network_force_close_socket(g_s_ftp.network->data_netc);
  287. network_release_ctrl(g_s_ftp.network->data_netc);
  288. g_s_ftp.network->data_netc = NULL;
  289. }
  290. break;
  291. case FTP_EVENT_CLOSE:
  292. g_s_ftp.is_run = 0;
  293. break;
  294. default:
  295. // LLOGE("ignore %x,%x,%x", event->ID, param, EV_NW_RESULT_EVENT);
  296. break;
  297. }
  298. return 0;
  299. }
  300. static int luat_ftp_pasv_connect(luat_ftp_ctrl_t *ftp_ctrl,uint32_t timeout_ms){
  301. char h1[4]={0},h2[4]={0},h3[4]={0},h4[4]={0},p1[4]={0},p2[4]={0},data_addr[64]={0};
  302. uint8_t port1,port2;
  303. uint16_t data_port;
  304. luat_ftp_cmd_send(&g_s_ftp, (uint8_t*)"PASV\r\n", strlen("PASV\r\n"),FTP_SOCKET_TIMEOUT);
  305. int ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  306. if (ret){
  307. return -1;
  308. }else{
  309. LLOGD("luat_ftp_pasv_connect cmd_recv_data",g_s_ftp.network->cmd_recv_data);
  310. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_ENTER_PASSIVE, 3)){
  311. LLOGD("ftp pasv_connect wrong");
  312. return -1;
  313. }
  314. }
  315. char *temp = memchr(g_s_ftp.network->cmd_recv_data, '(', strlen((const char *)(g_s_ftp.network->cmd_recv_data)));
  316. char *temp1 = memchr(temp+1, ',', strlen(temp)-1);
  317. memcpy(h1, temp+1, temp1-temp-1);
  318. char *temp2 = memchr(temp1+1, ',', strlen(temp1)-1);
  319. memcpy(h2, temp1+1, temp2-temp1-1);
  320. char *temp3 = memchr(temp2+1, ',', strlen(temp2)-1);
  321. memcpy(h3, temp2+1, temp3-temp2-1);
  322. char *temp4 = memchr(temp3+1, ',', strlen(temp3)-1);
  323. memcpy(h4, temp3+1, temp4-temp3-1);
  324. char *temp5 = memchr(temp4+1, ',', strlen(temp4)-1);
  325. memcpy(p1, temp4+1, temp5-temp4-1);
  326. char *temp6 = memchr(temp5+1, ')', strlen(temp5)-1);
  327. memcpy(p2, temp5+1, temp6-temp5-1);
  328. snprintf_(data_addr, 64, "%s.%s.%s.%s",h1,h2,h3,h4);
  329. port1 = (uint8_t)atoi(p1);
  330. port2 = (uint8_t)atoi(p2);
  331. data_port = port1 * 256 + port2;
  332. LLOGD("data_addr:%s data_port:%d",data_addr,data_port);
  333. if (memcmp(data_addr,"172.",4)==0||memcmp(data_addr,"192.",4)==0||memcmp(data_addr,"10.",3)==0||memcmp(data_addr,"127.0.0.1",9)==0||memcmp(data_addr,"169.254.0.0",11)==0||memcmp(data_addr,"169.254.0.16",12)==0){
  334. memset(data_addr,0,64);
  335. LLOGD("g_s_ftp.network->addr:%s",g_s_ftp.network->addr);
  336. memcpy(data_addr, g_s_ftp.network->addr, strlen(g_s_ftp.network->addr)+1);
  337. }
  338. LLOGD("data_addr:%s data_port:%d",data_addr,data_port);
  339. if (g_s_ftp.network->data_netc)
  340. {
  341. LLOGE("data_netc already create");
  342. return -1;
  343. }
  344. g_s_ftp.network->data_netc = network_alloc_ctrl(g_s_ftp.network->adapter_index);
  345. if (!g_s_ftp.network->data_netc){
  346. LLOGE("data_netc create fail");
  347. return -1;
  348. }
  349. network_init_ctrl(g_s_ftp.network->data_netc,NULL, luat_ftp_data_callback, g_s_ftp.network);
  350. network_set_base_mode(g_s_ftp.network->data_netc, 1, 10000, 0, 0, 0, 0);
  351. network_set_local_port(g_s_ftp.network->data_netc, 0);
  352. network_deinit_tls(g_s_ftp.network->data_netc);
  353. if(network_connect(g_s_ftp.network->data_netc, data_addr, strlen(data_addr), NULL, data_port, 0)<0){
  354. LLOGE("ftp data network connect fail");
  355. network_force_close_socket(g_s_ftp.network->data_netc);
  356. network_release_ctrl(g_s_ftp.network->data_netc);
  357. g_s_ftp.network->data_netc = NULL;
  358. return -1;
  359. }
  360. uint8_t is_timeout;
  361. OS_EVENT event;
  362. g_s_ftp.network->data_netc_connecting = 1;
  363. g_s_ftp.network->data_netc_online = 0;
  364. while(g_s_ftp.network->data_netc_connecting)
  365. {
  366. if (network_wait_event(g_s_ftp.network->cmd_netc, &event, timeout_ms, &is_timeout))
  367. {
  368. return -1;
  369. }
  370. if (is_timeout)
  371. {
  372. return -1;
  373. }
  374. if (event.ID)
  375. {
  376. ftp_task_cb(&event, NULL);
  377. }
  378. else
  379. {
  380. if (g_s_ftp.network->cmd_netc->new_rx_flag)
  381. {
  382. network_rx(g_s_ftp.network->cmd_netc, g_s_ftp.network->cmd_recv_data, 1024, 0, NULL, NULL, &g_s_ftp.network->cmd_recv_len);
  383. LLOGD("ftp cmd rx %.*s", g_s_ftp.network->cmd_recv_len, g_s_ftp.network->cmd_recv_data);
  384. }
  385. }
  386. }
  387. if (g_s_ftp.network->data_netc_online)
  388. {
  389. LLOGD("ftp pasv_connect ok");
  390. return 0;
  391. }
  392. return -1;
  393. }
  394. static int ftp_login(void)
  395. {
  396. int ret;
  397. if(network_connect(g_s_ftp.network->cmd_netc, g_s_ftp.network->addr, strlen(g_s_ftp.network->addr), NULL, g_s_ftp.network->port, FTP_SOCKET_TIMEOUT)){
  398. LLOGE("ftp network_connect fail");
  399. return -1;
  400. }
  401. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  402. if (ret){
  403. return -1;
  404. }else{
  405. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_SERVICE_NEW_OK, 3)){
  406. LLOGE("ftp connect error");
  407. return -1;
  408. }
  409. }
  410. LLOGD("ftp connect ok");
  411. memset(g_s_ftp.network->cmd_send_data,0,FTP_CMD_SEND_MAX);
  412. snprintf_((char *)(g_s_ftp.network->cmd_send_data), FTP_CMD_SEND_MAX, "USER %s\r\n",g_s_ftp.network->username);
  413. luat_ftp_cmd_send(&g_s_ftp, g_s_ftp.network->cmd_send_data, strlen((const char *)(g_s_ftp.network->cmd_send_data)),FTP_SOCKET_TIMEOUT);
  414. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  415. if (ret){
  416. LLOGE("ftp username wrong");
  417. return -1;
  418. }else{
  419. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_USERNAME_OK, 3)){
  420. LLOGE("ftp username wrong");
  421. return -1;
  422. }
  423. }
  424. LLOGD("ftp username ok");
  425. memset(g_s_ftp.network->cmd_send_data,0,FTP_CMD_SEND_MAX);
  426. snprintf_((char *)(g_s_ftp.network->cmd_send_data), FTP_CMD_SEND_MAX, "PASS %s\r\n",g_s_ftp.network->password);
  427. luat_ftp_cmd_send(&g_s_ftp, g_s_ftp.network->cmd_send_data, strlen((const char *)(g_s_ftp.network->cmd_send_data)),FTP_SOCKET_TIMEOUT);
  428. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  429. if (ret){
  430. LLOGE("ftp login wrong");
  431. return -1;
  432. }else{
  433. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_LOGIN_OK, 3)){
  434. LLOGE("ftp login wrong");
  435. return -1;
  436. }
  437. }
  438. LLOGD("ftp login ok");
  439. return 0;
  440. }
  441. static void ftp_task(void *param){
  442. int ret;
  443. int count = 0;
  444. luat_rtos_task_handle task_handle = g_s_ftp.task_handle;
  445. OS_EVENT task_event;
  446. rtos_msg_t msg = {0};
  447. uint8_t is_timeout = 0;
  448. g_s_ftp.is_run = 1;
  449. msg.handler = l_ftp_callback;
  450. luat_rtos_event_recv(g_s_ftp.task_handle, FTP_EVENT_LOGIN, &task_event, NULL, LUAT_WAIT_FOREVER);
  451. if (ftp_login())
  452. {
  453. LLOGE("ftp login fail");
  454. luat_ftp_close();
  455. msg.arg1 = 1;
  456. luat_msgbus_put(&msg, 0);
  457. g_s_ftp.task_handle = NULL;
  458. luat_rtos_task_delete(task_handle);
  459. return;
  460. }
  461. else
  462. {
  463. msg.arg1 = 0;
  464. msg.arg2 = 0;
  465. luat_msgbus_put(&msg, 0);
  466. }
  467. while (g_s_ftp.is_run) {
  468. is_timeout = 0;
  469. ret = network_wait_event(g_s_ftp.network->cmd_netc, &task_event, 3600000, &is_timeout);
  470. if (ret < 0)
  471. {
  472. LLOGE("ftp network error");
  473. goto wait_event_and_out;
  474. }
  475. else if (is_timeout || !task_event.ID)
  476. {
  477. if (g_s_ftp.network->cmd_netc->new_rx_flag)
  478. {
  479. network_rx(g_s_ftp.network->cmd_netc, g_s_ftp.network->cmd_recv_data, 1024, 0, NULL, NULL, &ret);
  480. LLOGD("ftp rx %dbyte", ret);
  481. }
  482. continue;
  483. }
  484. msg.arg1 = 0;
  485. msg.arg2 = 0;
  486. switch (task_event.ID)
  487. {
  488. case FTP_EVENT_LOGIN:
  489. break;
  490. case FTP_EVENT_PULL:
  491. if (g_s_ftp.network->data_netc)
  492. {
  493. network_force_close_socket(g_s_ftp.network->data_netc);
  494. network_release_ctrl(g_s_ftp.network->data_netc);
  495. g_s_ftp.network->data_netc = NULL;
  496. }
  497. if(luat_ftp_pasv_connect(&g_s_ftp,FTP_SOCKET_TIMEOUT)){
  498. LLOGE("ftp pasv_connect fail");
  499. goto operation_failed;
  500. }
  501. snprintf_((char *)(g_s_ftp.network->cmd_send_data), FTP_CMD_SEND_MAX, "RETR %s\r\n",g_s_ftp.network->remote_name);
  502. luat_ftp_cmd_send(&g_s_ftp, g_s_ftp.network->cmd_send_data, strlen((const char *)g_s_ftp.network->cmd_send_data),FTP_SOCKET_TIMEOUT);
  503. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  504. if (ret){
  505. goto operation_failed;
  506. }else{
  507. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_STATUS_OK, 3)){
  508. LLOGD("ftp RETR wrong");
  509. goto operation_failed;
  510. }
  511. }
  512. if (!g_s_ftp.network->data_netc_online)
  513. {
  514. g_s_ftp.network->cmd_recv_data[g_s_ftp.network->cmd_recv_len] = 0;
  515. LLOGD("ftp RETR maybe done!");
  516. if (strstr((const char *)(g_s_ftp.network->cmd_recv_data), "226 "))
  517. {
  518. LLOGD("ftp RETR ok!");
  519. if (g_s_ftp.fd){
  520. luat_fs_fclose(g_s_ftp.fd);
  521. g_s_ftp.fd = NULL;
  522. }
  523. luat_msgbus_put(&msg, 0);
  524. break;
  525. }
  526. }
  527. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  528. if (ret){
  529. goto operation_failed;
  530. }else{
  531. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_CLOSE_CONNECT, 3)){
  532. LLOGD("ftp RETR wrong");
  533. goto operation_failed;
  534. }
  535. }
  536. while (count<3 && g_s_ftp.network->data_netc_online!=0){
  537. luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT/3);
  538. count++;
  539. }
  540. if (g_s_ftp.fd){
  541. luat_fs_fclose(g_s_ftp.fd);
  542. g_s_ftp.fd = NULL;
  543. }
  544. luat_msgbus_put(&msg, 0);
  545. break;
  546. case FTP_EVENT_PUSH:
  547. if(luat_ftp_pasv_connect(&g_s_ftp,FTP_SOCKET_TIMEOUT)){
  548. LLOGD("ftp pasv_connect fail");
  549. goto operation_failed;
  550. }
  551. memset(g_s_ftp.network->cmd_send_data,0,FTP_CMD_SEND_MAX);
  552. snprintf_((char *)(g_s_ftp.network->cmd_send_data), FTP_CMD_SEND_MAX, "STOR %s\r\n",g_s_ftp.network->remote_name);
  553. luat_ftp_cmd_send(&g_s_ftp, g_s_ftp.network->cmd_send_data, strlen((const char *)g_s_ftp.network->cmd_send_data),FTP_SOCKET_TIMEOUT);
  554. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  555. if (ret){
  556. goto operation_failed;
  557. }else{
  558. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_STATUS_OK, 3)){
  559. LLOGD("ftp STOR wrong");
  560. goto operation_failed;
  561. }
  562. }
  563. uint8_t* buff = luat_heap_malloc(PUSH_BUFF_SIZE);
  564. int offset = 0;
  565. g_s_ftp.network->upload_done_size = 0;
  566. while (1) {
  567. memset(buff, 0, PUSH_BUFF_SIZE);
  568. int len = luat_fs_fread(buff, sizeof(uint8_t), PUSH_BUFF_SIZE, g_s_ftp.fd);
  569. if (len < 1)
  570. break;
  571. luat_ftp_data_send(&g_s_ftp, buff, len);
  572. offset += len;
  573. }
  574. luat_heap_free(buff);
  575. LLOGD("offset:%d file_size:%d",offset,g_s_ftp.network->local_file_size);
  576. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  577. if (g_s_ftp.network->upload_done_size != g_s_ftp.network->local_file_size)
  578. {
  579. LLOGE("upload not finish !!! %d,%d", g_s_ftp.network->upload_done_size, g_s_ftp.network->local_file_size);
  580. }
  581. if (ret){
  582. goto operation_failed;
  583. }else{
  584. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_CLOSE_CONNECT, 3)){
  585. LLOGD("ftp STOR wrong");
  586. }
  587. }
  588. while (count<3 && g_s_ftp.network->data_netc_online!=0){
  589. luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT/3);
  590. count++;
  591. }
  592. if (g_s_ftp.fd){
  593. luat_fs_fclose(g_s_ftp.fd);
  594. g_s_ftp.fd = NULL;
  595. }
  596. luat_msgbus_put(&msg, 0);
  597. break;
  598. case FTP_EVENT_CLOSE:
  599. g_s_ftp.is_run = 0;
  600. break;
  601. case FTP_EVENT_COMMAND:
  602. OS_DeInitBuffer(&g_s_ftp.result_buffer);
  603. if(!memcmp(g_s_ftp.network->cmd_send_data, "LIST", 4))
  604. {
  605. if(luat_ftp_pasv_connect(&g_s_ftp,FTP_SOCKET_TIMEOUT)){
  606. LLOGD("ftp pasv_connect fail");
  607. goto operation_failed;
  608. }
  609. luat_ftp_cmd_send(&g_s_ftp, g_s_ftp.network->cmd_send_data, strlen((const char *)(g_s_ftp.network->cmd_send_data)),FTP_SOCKET_TIMEOUT);
  610. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  611. if (ret){
  612. goto operation_failed;
  613. }else{
  614. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_STATUS_OK, 3)){
  615. LLOGE("ftp LIST wrong");
  616. goto operation_failed;
  617. }
  618. }
  619. if (!g_s_ftp.network->data_netc_online)
  620. {
  621. g_s_ftp.network->cmd_recv_data[g_s_ftp.network->cmd_recv_len] = 0;
  622. LLOGD("ftp LIST maybe done!");
  623. if (strstr((const char *)(g_s_ftp.network->cmd_recv_data), FTP_CLOSE_CONNECT))
  624. {
  625. LLOGD("ftp LIST ok!");
  626. msg.arg2 = 1;
  627. luat_msgbus_put(&msg, 0);
  628. break;
  629. }
  630. }
  631. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  632. if (ret){
  633. goto operation_failed;
  634. }else{
  635. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_CLOSE_CONNECT, 3)){
  636. LLOGE("ftp LIST wrong");
  637. goto operation_failed;
  638. }
  639. }
  640. msg.arg2 = 1;
  641. luat_msgbus_put(&msg, 0);
  642. break;
  643. }
  644. luat_ftp_cmd_send(&g_s_ftp, g_s_ftp.network->cmd_send_data, strlen((const char *)(g_s_ftp.network->cmd_send_data)),FTP_SOCKET_TIMEOUT);
  645. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  646. if (ret){
  647. goto operation_failed;
  648. }else{
  649. if (memcmp(g_s_ftp.network->cmd_send_data, "NOOP", 4)==0){
  650. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_COMMAND_OK, 3)){
  651. LLOGD("ftp COMMAND wrong");
  652. }
  653. }else if(memcmp(g_s_ftp.network->cmd_send_data, "TYPE", 4)==0){
  654. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_COMMAND_OK, 3)){
  655. LLOGD("ftp COMMAND wrong");
  656. }
  657. }else if(memcmp(g_s_ftp.network->cmd_send_data, "SYST", 4)==0){
  658. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_SYSTEM_TYPE, 3)){
  659. LLOGD("ftp COMMAND wrong");
  660. }
  661. }else if(memcmp(g_s_ftp.network->cmd_send_data, "PWD", 3)==0){
  662. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_PATHNAME_OK, 3)){
  663. LLOGD("ftp COMMAND wrong");
  664. }
  665. }else if(memcmp(g_s_ftp.network->cmd_send_data, "MKD", 3)==0){
  666. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_PATHNAME_OK, 3)){
  667. LLOGD("ftp COMMAND wrong");
  668. }
  669. }else if(memcmp(g_s_ftp.network->cmd_send_data, "CWD", 3)==0){
  670. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_REQUESTED_OK, 3)){
  671. LLOGD("ftp COMMAND wrong");
  672. }
  673. }else if(memcmp(g_s_ftp.network->cmd_send_data, "CDUP", 4)==0){
  674. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_REQUESTED_OK, 3)){
  675. LLOGD("ftp COMMAND wrong");
  676. }
  677. }else if(memcmp(g_s_ftp.network->cmd_send_data, "RMD", 3)==0){
  678. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_REQUESTED_OK, 3)){
  679. LLOGD("ftp COMMAND wrong");
  680. }
  681. }else if(memcmp(g_s_ftp.network->cmd_send_data, "DELE", 4)==0){
  682. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_FILE_REQUESTED_OK, 3)){
  683. LLOGD("ftp COMMAND wrong");
  684. }
  685. }else if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_DATA_CON_FAIL, 3)==0){
  686. LLOGD("ftp need pasv_connect");
  687. }
  688. }
  689. OS_BufferWrite(&g_s_ftp.result_buffer, g_s_ftp.network->cmd_recv_data, g_s_ftp.network->cmd_recv_len);
  690. msg.arg2 = 1;
  691. luat_msgbus_put(&msg, 0);
  692. break;
  693. default:
  694. break;
  695. }
  696. continue;
  697. operation_failed:
  698. msg.arg1 = 1;
  699. luat_msgbus_put(&msg, 0);
  700. }
  701. msg.arg1 = 0;
  702. msg.arg2 = 0;
  703. luat_ftp_cmd_send(&g_s_ftp, (uint8_t*)"QUIT\r\n", strlen("QUIT\r\n"),FTP_SOCKET_TIMEOUT);
  704. ret = luat_ftp_cmd_recv(&g_s_ftp,g_s_ftp.network->cmd_recv_data,&g_s_ftp.network->cmd_recv_len,FTP_SOCKET_TIMEOUT);
  705. if (ret){
  706. msg.arg1 = 1;
  707. }else{
  708. if (memcmp(g_s_ftp.network->cmd_recv_data, FTP_CLOSE_CONTROL, 3)){
  709. LLOGE("ftp QUIT wrong");
  710. msg.arg1 = 1;
  711. }
  712. }
  713. OS_BufferWrite(&g_s_ftp.result_buffer, g_s_ftp.network->cmd_recv_data, g_s_ftp.network->cmd_recv_len);
  714. msg.arg2 = 1;
  715. luat_ftp_close();
  716. luat_msgbus_put(&msg, 0);
  717. g_s_ftp.task_handle = NULL;
  718. luat_rtos_task_delete(task_handle);
  719. return;
  720. wait_event_and_out:
  721. while(1)
  722. {
  723. luat_rtos_event_recv(g_s_ftp.task_handle, 0, &task_event, NULL, LUAT_WAIT_FOREVER);
  724. if (task_event.ID >= FTP_EVENT_LOGIN && task_event.ID <= FTP_EVENT_CLOSE)
  725. {
  726. luat_ftp_close();
  727. msg.arg1 = 1;
  728. luat_msgbus_put(&msg, 0);
  729. g_s_ftp.task_handle = NULL;
  730. luat_rtos_task_delete(task_handle);
  731. return;
  732. }
  733. }
  734. }
  735. /*
  736. FTP客户端
  737. @api ftp.login(adapter,ip_addr,port,username,password)
  738. @int 适配器序号, 只能是socket.ETH0, socket.STA, socket.AP,如果不填,会选择平台自带的方式,然后是最后一个注册的适配器
  739. @string ip_addr 地址
  740. @string port 端口,默认21
  741. @string username 用户名
  742. @string password 密码
  743. @bool/table 是否为ssl加密连接,默认不加密,true为无证书最简单的加密,table为有证书的加密 <br>server_cert 服务器ca证书数据 <br>client_cert 客户端ca证书数据 <br>client_key 客户端私钥加密数据 <br>client_password 客户端私钥口令数据
  744. @return bool/string 成功返回true 失败返回string
  745. @usage
  746. ftp_login = ftp.login(nil,"xxx")
  747. */
  748. static int l_ftp_login(lua_State *L) {
  749. size_t server_cert_len,client_cert_len, client_key_len, client_password_len;
  750. size_t len = 0;
  751. const char *server_cert = NULL;
  752. const char *client_cert = NULL;
  753. const char *client_key = NULL;
  754. const char *client_password = NULL;
  755. const char *username = NULL;
  756. const char *password = NULL;
  757. int result = 0;
  758. // mbedtls_debug_set_threshold(4);
  759. if (g_s_ftp.network){
  760. LLOGE("ftp already login, please close first");
  761. result = FTP_ERROR_STATE;
  762. goto error;
  763. }
  764. g_s_ftp.network = (luat_ftp_network_t *)luat_heap_malloc(sizeof(luat_ftp_network_t));
  765. if (!g_s_ftp.network){
  766. LLOGE("out of memory when malloc g_s_ftp.network");
  767. result = FTP_ERROR_NO_MEM;
  768. goto error;
  769. }
  770. memset(g_s_ftp.network, 0, sizeof(luat_ftp_network_t));
  771. g_s_ftp.network->adapter_index = luaL_optinteger(L, 1, network_get_last_register_adapter());
  772. if (g_s_ftp.network->adapter_index >= NW_ADAPTER_QTY){
  773. LLOGE("bad network adapter index %d", g_s_ftp.network->adapter_index);
  774. result = FTP_ERROR_STATE;
  775. goto error;
  776. }
  777. g_s_ftp.network->cmd_netc = network_alloc_ctrl(g_s_ftp.network->adapter_index);
  778. if (!g_s_ftp.network->cmd_netc){
  779. LLOGE("cmd_netc create fail");
  780. result = FTP_ERROR_NO_MEM;
  781. goto error;
  782. }
  783. const char *addr = luaL_checklstring(L, 2, &len);
  784. if (len > 0 && len < 64)
  785. memcpy(g_s_ftp.network->addr, addr, len + 1);
  786. g_s_ftp.network->port = luaL_optinteger(L, 3, 21);
  787. username = luaL_optlstring(L, 4, "",&len);
  788. if (len > 0 && len < 64)
  789. memcpy(g_s_ftp.network->username, username, len + 1);
  790. password = luaL_optlstring(L, 5, "",&len);
  791. if (len > 0 && len < 64)
  792. memcpy(g_s_ftp.network->password, password, len + 1);
  793. // 加密相关
  794. if (lua_isboolean(L, 6)){
  795. g_s_ftp.network->is_tls = lua_toboolean(L, 6);
  796. }
  797. if (lua_istable(L, 6)){
  798. g_s_ftp.network->is_tls = 1;
  799. lua_pushstring(L, "server_cert");
  800. if (LUA_TSTRING == lua_gettable(L, 6)) {
  801. server_cert = luaL_checklstring(L, -1, &server_cert_len);
  802. }
  803. lua_pop(L, 1);
  804. lua_pushstring(L, "client_cert");
  805. if (LUA_TSTRING == lua_gettable(L, 6)) {
  806. client_cert = luaL_checklstring(L, -1, &client_cert_len);
  807. }
  808. lua_pop(L, 1);
  809. lua_pushstring(L, "client_key");
  810. if (LUA_TSTRING == lua_gettable(L, 6)) {
  811. client_key = luaL_checklstring(L, -1, &client_key_len);
  812. }
  813. lua_pop(L, 1);
  814. lua_pushstring(L, "client_password");
  815. if (LUA_TSTRING == lua_gettable(L, 6)) {
  816. client_password = luaL_checklstring(L, -1, &client_password_len);
  817. }
  818. lua_pop(L, 1);
  819. }
  820. if (g_s_ftp.network->is_tls){
  821. if (lua_isstring(L, 6)){
  822. server_cert = luaL_checklstring(L, 6, &server_cert_len);
  823. }
  824. if (lua_isstring(L, 7)){
  825. client_cert = luaL_checklstring(L, 7, &client_cert_len);
  826. }
  827. if (lua_isstring(L, 8)){
  828. client_key = luaL_checklstring(L, 8, &client_key_len);
  829. }
  830. if (lua_isstring(L, 9)){
  831. client_password = luaL_checklstring(L, 9, &client_password_len);
  832. }
  833. if (network_init_tls(g_s_ftp.network->cmd_netc, (server_cert || client_cert)?2:0)){
  834. result = FTP_ERROR_CLOSE;
  835. goto error;
  836. }
  837. if (server_cert){
  838. network_set_server_cert(g_s_ftp.network->cmd_netc, (const unsigned char *)server_cert, server_cert_len+1);
  839. }
  840. if (client_cert){
  841. network_set_client_cert(g_s_ftp.network->cmd_netc, (const unsigned char *)client_cert, client_cert_len+1,
  842. (const unsigned char *)client_key, client_key_len+1,
  843. (const unsigned char *)client_password, client_password_len+1);
  844. }
  845. }else{
  846. network_deinit_tls(g_s_ftp.network->cmd_netc);
  847. }
  848. network_set_ip_invaild(&g_s_ftp.network->ip_addr);
  849. luat_rtos_task_create(&g_s_ftp.task_handle, 8*1024, 10, "ftp", ftp_task, NULL, 16);
  850. network_init_ctrl(g_s_ftp.network->cmd_netc,g_s_ftp.task_handle, ftp_task_cb, NULL);
  851. network_set_base_mode(g_s_ftp.network->cmd_netc, 1, 30000, 0, 0, 0, 0);
  852. network_set_local_port(g_s_ftp.network->cmd_netc, 0);
  853. g_s_ftp.idp = luat_pushcwait(L);
  854. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_LOGIN, 0, 0, 0, LUAT_WAIT_FOREVER);
  855. return 1;
  856. error:
  857. LLOGE("ftp login fail");
  858. luat_ftp_close();
  859. lua_pushinteger(L, result);
  860. luat_pushcwait_error(L,1);
  861. return 1;
  862. }
  863. /*
  864. FTP命令
  865. @api ftp.command(cmd)
  866. @string cmd 命令 目前支持:NOOP SYST TYPE PWD MKD CWD CDUP RMD DELE LIST
  867. @return string 成功返回true 失败返回string
  868. @usage
  869. print(ftp.command("NOOP").wait())
  870. print(ftp.command("SYST").wait())
  871. print(ftp.command("TYPE I").wait())
  872. print(ftp.command("PWD").wait())
  873. print(ftp.command("MKD QWER").wait())
  874. print(ftp.command("CWD /QWER").wait())
  875. print(ftp.command("CDUP").wait())
  876. print(ftp.command("RMD QWER").wait())
  877. print(ftp.command("DELE /1/12222.txt").wait())
  878. */
  879. static int l_ftp_command(lua_State *L) {
  880. if (!g_s_ftp.network){
  881. LLOGE("please login first");
  882. goto error;
  883. }
  884. g_s_ftp.idp = luat_pushcwait(L);
  885. size_t len;
  886. const char *cmd = luaL_checklstring(L, 1, &len);
  887. if (memcmp(cmd, "NOOP", 4)==0){
  888. LLOGD("command: NOOP");
  889. }else if(memcmp(cmd, "SYST", 4)==0){
  890. LLOGD("command: SYST");
  891. }else if(memcmp(cmd, "MKD", 3)==0){
  892. LLOGD("command: MKD");
  893. }else if(memcmp(cmd, "CWD", 3)==0){
  894. LLOGD("command: CWD");
  895. }else if(memcmp(cmd, "CDUP", 4)==0){
  896. LLOGD("command: CDUP");
  897. }else if(memcmp(cmd, "RMD", 3)==0){
  898. LLOGD("command: RMD");
  899. }else if(memcmp(cmd, "PWD", 3)==0){
  900. LLOGD("command: RMD");
  901. }else if(memcmp(cmd, "DELE", 4)==0){
  902. LLOGD("command: DELE");
  903. }else if(memcmp(cmd, "TYPE", 4)==0){
  904. LLOGD("command: TYPE");
  905. }else if(memcmp(cmd, "LIST", 4)==0){
  906. LLOGD("command: LIST");
  907. }else{
  908. LLOGE("not support cmd:%s",cmd);
  909. lua_pushinteger(L,FTP_ERROR_FILE);
  910. luat_pushcwait_error(L,1);
  911. return 1;
  912. }
  913. memset(g_s_ftp.network->cmd_send_data,0,FTP_CMD_SEND_MAX);
  914. snprintf_((char *)(g_s_ftp.network->cmd_send_data), FTP_CMD_SEND_MAX, "%s\r\n",cmd);
  915. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_COMMAND, (uint32_t)g_s_ftp.idp, (uint32_t)(g_s_ftp.idp >> 32), 0, LUAT_WAIT_FOREVER);
  916. return 1;
  917. error:
  918. LLOGE("ftp command fail");
  919. lua_pushinteger(L,FTP_ERROR_FILE);
  920. luat_pushcwait_error(L,1);
  921. return 1;
  922. }
  923. /*
  924. FTP文件下载
  925. @api ftp.pull(local_name,remote_name)
  926. @string local_name 本地文件
  927. @string remote_name 服务器文件
  928. @return bool/string 成功返回true 失败返回string
  929. @usage
  930. ftp.pull("/1222.txt","/1222.txt").wait()
  931. */
  932. static int l_ftp_pull(lua_State *L) {
  933. size_t len;
  934. if (!g_s_ftp.network){
  935. LLOGE("please login first");
  936. goto error;
  937. }
  938. g_s_ftp.idp = luat_pushcwait(L);
  939. const char * local_name = luaL_optlstring(L, 1, "",&len);
  940. luat_fs_remove(local_name);
  941. if (g_s_ftp.fd)
  942. {
  943. luat_fs_fclose(g_s_ftp.fd);
  944. g_s_ftp.fd = NULL;
  945. }
  946. g_s_ftp.fd = luat_fs_fopen(local_name, "wb+");
  947. if (g_s_ftp.fd == NULL) {
  948. LLOGE("open download file fail %s", local_name);
  949. goto error;
  950. }
  951. g_s_ftp.network->local_file_size = luat_fs_fsize(local_name);
  952. const char * remote_name = luaL_optlstring(L, 2, "",&len);
  953. memcpy(g_s_ftp.network->remote_name, remote_name, len + 1);
  954. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_PULL, 0, 0, 0, 0);
  955. return 1;
  956. error:
  957. LLOGE("ftp pull fail");
  958. lua_pushinteger(L,FTP_ERROR_FILE);
  959. luat_pushcwait_error(L,1);
  960. return 1;
  961. }
  962. /*
  963. FTP文件上传
  964. @api ftp.push(local_name,remote_name)
  965. @string local_name 本地文件
  966. @string remote_name 服务器文件
  967. @return bool/string 成功返回true 失败返回string
  968. @usage
  969. ftp.push("/1222.txt","/1222.txt").wait()
  970. */
  971. static int l_ftp_push(lua_State *L) {
  972. size_t len;
  973. if (!g_s_ftp.network){
  974. LLOGE("please login first");
  975. goto error;
  976. }
  977. g_s_ftp.idp = luat_pushcwait(L);
  978. const char * local_name = luaL_optlstring(L, 1, "",&len);
  979. if (g_s_ftp.fd)
  980. {
  981. luat_fs_fclose(g_s_ftp.fd);
  982. g_s_ftp.fd = NULL;
  983. }
  984. g_s_ftp.fd = luat_fs_fopen(local_name, "rb");
  985. if (g_s_ftp.fd == NULL) {
  986. LLOGE("open download file fail %s", local_name);
  987. goto error;
  988. }
  989. g_s_ftp.network->local_file_size = luat_fs_fsize(local_name);
  990. const char * remote_name = luaL_optlstring(L, 2, "",&len);
  991. memcpy(g_s_ftp.network->remote_name, remote_name, len + 1);
  992. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_PUSH, 0, 0, 0, LUAT_WAIT_FOREVER);
  993. return 1;
  994. error:
  995. LLOGE("ftp push fail");
  996. lua_pushinteger(L,FTP_ERROR_CONNECT);
  997. luat_pushcwait_error(L,1);
  998. return 1;
  999. }
  1000. /*
  1001. FTP客户端关闭
  1002. @api ftp.close()
  1003. @return bool/string 成功返回true 失败返回string
  1004. @usage
  1005. ftp.close().wait()
  1006. */
  1007. static int l_ftp_close(lua_State *L) {
  1008. if (!g_s_ftp.network){
  1009. LLOGE("please login first");
  1010. goto error;
  1011. }
  1012. g_s_ftp.idp = luat_pushcwait(L);
  1013. luat_rtos_event_send(g_s_ftp.task_handle, FTP_EVENT_CLOSE, 0, 0, 0, LUAT_WAIT_FOREVER);
  1014. return 1;
  1015. error:
  1016. lua_pushinteger(L,FTP_ERROR_CONNECT);
  1017. luat_pushcwait_error(L,1);
  1018. return 1;
  1019. }
  1020. #include "rotable2.h"
  1021. #ifdef LUAT_USE_NETWORK
  1022. static const rotable_Reg_t reg_ftp[] =
  1023. {
  1024. {"login", ROREG_FUNC(l_ftp_login)},
  1025. {"command", ROREG_FUNC(l_ftp_command)},
  1026. {"pull", ROREG_FUNC(l_ftp_pull)},
  1027. {"push", ROREG_FUNC(l_ftp_push)},
  1028. {"close", ROREG_FUNC(l_ftp_close)},
  1029. { NULL, ROREG_INT(0)}
  1030. };
  1031. #else
  1032. static const rotable_Reg_t reg_ftp_emtry[] =
  1033. {
  1034. { NULL, ROREG_INT(0)}
  1035. };
  1036. #endif
  1037. LUAMOD_API int luaopen_ftp( lua_State *L ) {
  1038. #ifdef LUAT_USE_NETWORK
  1039. luat_newlib2(L, reg_ftp);
  1040. #else
  1041. luat_newlib2(L, reg_ftp_emtry);
  1042. LLOGE("ftp require network enable!!");
  1043. #endif
  1044. return 1;
  1045. }