294 lines
7.8 KiB
C
Raw Normal View History

2022-11-10 16:55:14 +08:00
/*
* Copyright (c) 2006-2022, RT-Thread Development Team
*
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date Author Notes
* 2017-07-26 chenyong modify log information
*/
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
2022-11-10 16:55:14 +08:00
#include <string.h>
#include "pika_adapter_rtt.h"
#include "webclient.h"
#ifdef RT_USING_DFS
#include <fcntl.h>
#include <unistd.h>
2022-11-10 16:55:14 +08:00
#define DBG_ENABLE
#define DBG_SECTION_NAME "web.file"
2022-11-10 16:55:14 +08:00
#ifdef WEBCLIENT_DEBUG
#define DBG_LEVEL DBG_LOG
2022-11-10 16:55:14 +08:00
#else
#define DBG_LEVEL DBG_INFO
2022-11-10 16:55:14 +08:00
#endif /* WEBCLIENT_DEBUG */
#define DBG_COLOR
#include <rtdbg.h>
/**
* send GET request and store response data into the file.
*
* @param URI input server address
* @param filename store response date to filename
*
* @return <0: GET request failed
* =0: success
*/
int webclient_get_file(const char* URI, const char* filename) {
2022-11-10 16:55:14 +08:00
int fd = -1, rc = WEBCLIENT_OK;
size_t offset;
int length, total_length = 0;
unsigned char* ptr = RT_NULL;
2022-11-10 16:55:14 +08:00
struct webclient_session* session = RT_NULL;
int resp_status = 0;
session = webclient_session_create(WEBCLIENT_HEADER_BUFSZ);
if (session == RT_NULL) {
2022-11-10 16:55:14 +08:00
rc = -WEBCLIENT_NOMEM;
goto __exit;
}
if ((resp_status = webclient_get(session, URI)) != 200) {
LOG_E("get file failed, wrong response: %d (-0x%X).", resp_status,
resp_status);
2022-11-10 16:55:14 +08:00
rc = -WEBCLIENT_ERROR;
goto __exit;
}
fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0);
if (fd < 0) {
2022-11-10 16:55:14 +08:00
LOG_E("get file failed, open file(%s) error.", filename);
rc = -WEBCLIENT_ERROR;
goto __exit;
}
ptr = (unsigned char*)web_malloc(WEBCLIENT_RESPONSE_BUFSZ);
if (ptr == RT_NULL) {
2022-11-10 16:55:14 +08:00
LOG_E("get file failed, no memory for response buffer.");
rc = -WEBCLIENT_NOMEM;
goto __exit;
}
if (session->content_length < 0) {
while (1) {
2022-11-10 16:55:14 +08:00
length = webclient_read(session, ptr, WEBCLIENT_RESPONSE_BUFSZ);
if (length > 0) {
2022-11-10 16:55:14 +08:00
write(fd, ptr, length);
total_length += length;
LOG_RAW(">");
} else {
2022-11-10 16:55:14 +08:00
break;
}
}
} else {
for (offset = 0; offset < (size_t)session->content_length;) {
length = webclient_read(
session, ptr,
session->content_length - offset > WEBCLIENT_RESPONSE_BUFSZ
? WEBCLIENT_RESPONSE_BUFSZ
: session->content_length - offset);
if (length > 0) {
2022-11-10 16:55:14 +08:00
write(fd, ptr, length);
total_length += length;
LOG_RAW(">");
} else {
2022-11-10 16:55:14 +08:00
break;
}
offset += length;
}
}
if (total_length) {
2022-11-10 16:55:14 +08:00
LOG_D("save %d bytes.", total_length);
}
__exit:
if (fd >= 0) {
2022-11-10 16:55:14 +08:00
close(fd);
}
if (session != RT_NULL) {
2022-11-10 16:55:14 +08:00
webclient_close(session);
}
if (ptr != RT_NULL) {
2022-11-10 16:55:14 +08:00
web_free(ptr);
}
return rc;
}
/**
* post file to http server.
*
* @param URI input server address
* @param filename post data filename
* @param form_data form data
*
* @return <0: POST request failed
* =0: success
*/
int webclient_post_file(const char* URI,
const char* filename,
const char* form_data) {
2022-11-10 16:55:14 +08:00
size_t length;
char boundary[60];
int fd = -1, rc = WEBCLIENT_OK;
char *header = RT_NULL, *header_ptr;
unsigned char *buffer = RT_NULL, *buffer_ptr;
struct webclient_session* session = RT_NULL;
int resp_data_len = 0;
fd = open(filename, O_RDONLY, 0);
if (fd < 0) {
2022-11-10 16:55:14 +08:00
LOG_D("post file failed, open file(%s) error.", filename);
rc = -WEBCLIENT_FILE_ERROR;
goto __exit;
}
/* get the size of file */
length = lseek(fd, 0, SEEK_END);
lseek(fd, 0, SEEK_SET);
buffer = (unsigned char*)web_calloc(1, WEBCLIENT_RESPONSE_BUFSZ);
if (buffer == RT_NULL) {
2022-11-10 16:55:14 +08:00
LOG_D("post file failed, no memory for response buffer.");
rc = -WEBCLIENT_NOMEM;
goto __exit;
}
header = (char*)web_calloc(1, WEBCLIENT_HEADER_BUFSZ);
if (header == RT_NULL) {
2022-11-10 16:55:14 +08:00
LOG_D("post file failed, no memory for header buffer.");
rc = -WEBCLIENT_NOMEM;
goto __exit;
}
header_ptr = header;
/* build boundary */
web_snprintf(boundary, sizeof(boundary),
"----------------------------%012d", rt_tick_get());
2022-11-10 16:55:14 +08:00
/* build encapsulated mime_multipart information*/
buffer_ptr = buffer;
/* first boundary */
buffer_ptr += web_snprintf((char*)buffer_ptr,
WEBCLIENT_RESPONSE_BUFSZ - (buffer_ptr - buffer),
"--%s\r\n", boundary);
buffer_ptr += web_snprintf(
(char*)buffer_ptr, WEBCLIENT_RESPONSE_BUFSZ - (buffer_ptr - buffer),
"Content-Disposition: form-data; %s\r\n", form_data);
buffer_ptr += web_snprintf(
(char*)buffer_ptr, WEBCLIENT_RESPONSE_BUFSZ - (buffer_ptr - buffer),
"Content-Type: application/octet-stream\r\n\r\n");
2022-11-10 16:55:14 +08:00
/* calculate content-length */
length += buffer_ptr - buffer;
length += strlen(boundary) + 8; /* add the last boundary */
/* build header for upload */
header_ptr +=
web_snprintf(header_ptr, WEBCLIENT_HEADER_BUFSZ - (header_ptr - header),
"Content-Length: %d\r\n", length);
header_ptr += web_snprintf(
header_ptr, WEBCLIENT_HEADER_BUFSZ - (header_ptr - header),
"Content-Type: multipart/form-data; boundary=%s\r\n", boundary);
2022-11-10 16:55:14 +08:00
session = webclient_session_create(WEBCLIENT_HEADER_BUFSZ);
if (session == RT_NULL) {
2022-11-10 16:55:14 +08:00
rc = -WEBCLIENT_NOMEM;
goto __exit;
}
strncpy(session->header->buffer, header, strlen(header));
session->header->length = strlen(session->header->buffer);
rc = webclient_post(session, URI, NULL, 0);
if (rc < 0) {
2022-11-10 16:55:14 +08:00
goto __exit;
}
/* send mime_multipart */
webclient_write(session, buffer, buffer_ptr - buffer);
/* send file data */
while (1) {
2022-11-10 16:55:14 +08:00
length = read(fd, buffer, WEBCLIENT_RESPONSE_BUFSZ);
if (length <= 0) {
2022-11-10 16:55:14 +08:00
break;
}
webclient_write(session, buffer, length);
}
/* send last boundary */
web_snprintf((char*)buffer, WEBCLIENT_RESPONSE_BUFSZ, "\r\n--%s--\r\n",
boundary);
2022-11-10 16:55:14 +08:00
webclient_write(session, buffer, strlen(boundary) + 8);
extern int webclient_handle_response(struct webclient_session * session);
if (webclient_handle_response(session) != 200) {
2022-11-10 16:55:14 +08:00
rc = -WEBCLIENT_ERROR;
goto __exit;
}
resp_data_len = webclient_content_length_get(session);
if (resp_data_len > 0) {
2022-11-10 16:55:14 +08:00
int bytes_read = 0;
web_memset(buffer, 0x00, WEBCLIENT_RESPONSE_BUFSZ);
do {
2022-11-10 16:55:14 +08:00
bytes_read = webclient_read(session, buffer,
resp_data_len < WEBCLIENT_RESPONSE_BUFSZ
? resp_data_len
: WEBCLIENT_RESPONSE_BUFSZ);
if (bytes_read <= 0) {
2022-11-10 16:55:14 +08:00
break;
}
resp_data_len -= bytes_read;
} while (resp_data_len > 0);
2022-11-10 16:55:14 +08:00
}
__exit:
if (fd >= 0) {
2022-11-10 16:55:14 +08:00
close(fd);
}
if (session != RT_NULL) {
2022-11-10 16:55:14 +08:00
webclient_close(session);
}
if (buffer != RT_NULL) {
2022-11-10 16:55:14 +08:00
web_free(buffer);
}
if (header != RT_NULL) {
2022-11-10 16:55:14 +08:00
web_free(header);
}
return rc;
}
int wget(int argc, char** argv) {
if (argc != 3) {
2022-11-10 16:55:14 +08:00
rt_kprintf("Please using: wget <URI> <filename>\n");
return -1;
}
webclient_get_file(argv[1], argv[2]);
return 0;
}
#ifdef FINSH_USING_MSH
#include <finsh.h>
MSH_CMD_EXPORT(wget, Get file by URI : wget<URI><filename>.);
2022-11-10 16:55:14 +08:00
#endif /* FINSH_USING_MSH */
#endif /* RT_USING_DFS */