⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ftp.c

📁 linux下流媒体下载程序代码
💻 C
📖 第 1 页 / 共 2 页
字号:
    }    else if(ss.ss_family == AF_INET6) {	struct sockaddr_in6 *sin6p;	char local_ipv6addr[INET6_ADDRSTRLEN + 4];		sin6p = (struct sockaddr_in6 *)&ss;	inet_ntop(sin6p->sin6_family,&(sin6p->sin6_addr.s6_addr),		  local_ipv6addr,INET6_ADDRSTRLEN + 4);		snprintf(sendbuffer,sizeof(sendbuffer) - 1,		 "EPRT |2|%s|%d|\n",local_ipv6addr,local_port);	    }    else {	display(MSDL_ERR,"unknown protocol family: %d\n",		ss.ss_family);	*newsock = 0;	return -1;    }        ftp_send_command(stream,sendbuffer);        *newsock = data_wait_sock;        return ftp_recv_response_ignore_message(stream);}/* * switch to ftp binary mode *             return value:     status code */static int ftp_bin_mode(struct stream_t *stream){    ftp_send_command(stream,"TYPE I\n");    return ftp_recv_response_ignore_message(stream);}/* * seek to requested position using REST request * set dlopts->byterange *             return value:     1 : success *                              -1 : no file name specified, or no local file */static int ftp_prepare_resuming(struct stream_t *stream){    uint64_t filesize = 0;    int ret = 0;        /*     * find same file name     */        ret = get_filesize(stream->localfile,&filesize);    if(ret < 0) {	display(MSDL_ERR,		"ftp resume: no such file \"%s\", not resuming\n",stream->localfile);	goto failed;    }    if(stream->dlopts->byterange) { /* free old byterange */	free(stream->dlopts->byterange);    }        stream->dlopts->byterange = make_byterange_from_filesize(filesize);    stream->resumeinfo->resume_start_offset = filesize;    display(MSDL_DBG,	    "ftp resume: start pos: %lld [%llx]\n",	    filesize,filesize);        return 1;  failed:    stream->resumeinfo->resume_start_offset = 0;    stream->resumeinfo->resume_req_success = 0;    return -1;}/* * ftp seek by REST req, from stream->dlopts->byterange * return value:   status code ... network transfer OK (does not mean success) *                          -1 ... error, or did nothing */static int ftp_seek_pos(struct stream_t *stream){    if(stream->dlopts->byterange) {	struct ftp_response_t *ftp_response = NULL;	char sendbuffer[128]; /* this is enough */	int status_code = 0;	uint64_t begin = 0,end = 0;	char *reason = NULL;	int byterange_valid = 0;	byterange_valid = ftp_interpret_byterange(stream->dlopts->byterange,&begin,&end,&reason);	if(!byterange_valid) {	    display(MSDL_ERR,		    "ftp_seek_pos() error: range string \"%s\" not valid\n"		    "%s\n",stream->dlopts->byterange,reason);	    return 0;	}		snprintf(sendbuffer,127,"REST %llu\n",(unsigned long long)begin);	ftp_send_command(stream,sendbuffer);		ftp_response = new_ftp_response_t();	status_code = ftp_recv_response(stream,ftp_response);	if(is_ftp_OK(status_code)) {	    display(MSDL_DBG,"seek OK\n");	    if(end && (end > begin)) { /* finish offset sepcified and length > 0 */		stream->stream_ctrl->ftp_ctrl->transfer_force_end_size = end - begin;	    }	}	else {	    display(MSDL_ERR,"ftp REST failed:\n");	    if(ftp_response->lines) {		display(MSDL_ERR,			"%s\n",(char *)ftp_response->lines->p);	    }	}	free_ftp_response_t(ftp_response);	return status_code;    }    return -1;}/* * ask file size of 'filepath' and it goes to 'size' *             return value:     status code */static int ftp_ask_size(struct stream_t *stream,const char *filepath,uint64_t *size){    struct ftp_response_t *ftp_response;    char *sendbuffer = xmalloc(strlen(filepath) + 16);    int status_code;        snprintf(sendbuffer,BUFSIZE_1K,"SIZE %s\n",filepath);    ftp_send_command(stream,sendbuffer);    ftp_response = new_ftp_response_t();    status_code = ftp_recv_response(stream,ftp_response);    if(is_ftp_OK(status_code)) {	char *p = (char *)ftp_response->lines->p;	uint64_t filesize = 0;		while(isdigit(*p)) p++; /* skip status code */	while((*p) == ' ') p++; /* skip white space */	while(isdigit(*p)) {	    filesize *= 10;	    filesize += *p - '0';	    p++;	}	display(MSDL_DBG,"file size %d bytes\n",filesize);		*size = filesize;    }    else {	if(ftp_response->lines) {	    display(MSDL_ERR,"%s",(char *)ftp_response->lines->p);	}	*size = 0;    }    free_ftp_response_t(ftp_response);    free(sendbuffer);    return status_code;}/* * read ftp byterange [0-9]+ * return value: 1 ... valid  0 ... invalid */static int ftp_byterange_read(const char *str,uint64_t *value,char **reason_ret){    const char *p = str;    uint64_t val = 0;    for(; *p != '\0' ;p++) {	if(!isdigit(*p)) {	    if(reason_ret) {		*reason_ret = "invalid character in range string";	    }	    *value = 0;	    return 0;	}	val *= 10;	val += *p - '0';    }        *reason_ret = NULL;    *value = val;    return 1;}/* * intepret range xxx-xxx *  * return value:    1 ... interpreted successfully *                  0 ... string error (not valid str) */static int ftp_interpret_byterange(const char *str,uint64_t *begin,uint64_t *end,				   char **reason_ret){    char *p = NULL;    int valid = 0;    char *reason = NULL;    p = strchr(str,'-');    if(p == NULL) { /* have to have exactly one '-' */	reason = "must use \'-\' to tell range";	if(reason_ret) {	    *reason_ret = reason;	}	goto failed;    }    else if(p == str) { /* ( "-" npt-time ) */	valid = ftp_byterange_read(p + 1,end,reason_ret);	if(!valid) {	    goto failed;	}	return valid; /* 1 */    }    else { /* 00.23- */	int start_str_len = p - str;	char *range_str = NULL;		if(*(p+1) != '\0') { /* not 00.23- */	    valid = ftp_byterange_read(p + 1,end,reason_ret);	    if(!valid) {		goto failed;	    }	}	/* end string OK */	/* check start string */	range_str = xmalloc(start_str_len + 1);	strncpy(range_str,str,start_str_len);	range_str[start_str_len] = '\0';	valid = ftp_byterange_read(range_str,begin,reason_ret);	free(range_str);	if(!valid) {	    goto failed;	}	return valid;    }      failed:    *begin = 0;    *end = 0;    return 0;}/* * send RETR request for downloading file. */static int ftp_retr(struct stream_t *stream,const char *filepath){    char *sendbuffer = xmalloc(strlen(filepath) + 16);    struct ftp_response_t *ftp_response;    int status_code;    snprintf(sendbuffer,BUFSIZE_1K,"RETR %s\n",filepath);    ftp_send_command(stream,sendbuffer);        ftp_response = new_ftp_response_t();    status_code = ftp_recv_response(stream,ftp_response);    if(!is_ftp_OK(status_code)) {	display(MSDL_ERR,"%s",(char *)ftp_response->lines->p);    }    free_ftp_response_t(ftp_response);    free(sendbuffer);    return status_code;}/* * starts ftp streaming(actually this is downlaoding). *  *    return value :   negative or 0  ... error *                                 1  ... success */int ftp_streaming_start(struct stream_t *stream){    struct stream_ctrl_t *stream_ctrl = stream->stream_ctrl;    struct url_t *url = stream->url;    struct download_opts_t *dlopts = stream->dlopts;        int command_sock = 0;    int data_wait_sock = 0;    int data_sock = 0;        int status_code = 0;    uint64_t file_size = 0;        stream_ctrl->status = STREAMING_HANDSHAKING;        if(stream->dlopts->no_passive_ftp) {	stream->stream_ctrl->ftp_ctrl->mode = ACTIVE_FTP;    }        set_serverinfo(stream->serverinfo,url->hostname,url->port,NULL,0,FTP_PORT);        command_sock = server_connect(stream->serverinfo->connect_host,				  stream->serverinfo->connect_port);    if(command_sock < 0) {	goto failed;    }        /*      stream_sock is CURRENT using socket, so now it's command_sock      ( later changed to data_sock, as it will be active )    */    stream->netsock->sock = command_sock;    stream_ctrl->ftp_ctrl->command_sock = command_sock;    status_code = ftp_read_welcome(stream);    if(!is_ftp_OK(status_code)) {	goto failed;    }        /*      USER, default is anonymous login.      priority:  url->username --> dlopts->username --> "anonymous"    */    status_code = ftp_user(stream,			   (url->username) ? url->username : 			   ((dlopts->username) ? dlopts->username : 			    "anonymous"));        if(status_code != 230) {	/*	  PASS (don't need to send this if user was already logged in with USER)	*/	status_code =ftp_pass(stream,			      (url->password) ? url->password :			      ((dlopts->password) ? dlopts->password :			       ftp_default_passwd));	if(!is_ftp_OK(status_code)) {	    goto failed;	}    }        /*      PASV (if not ACTIVE mode specified)            this may fail, and in that case fall back to ACTIVE.    */    if(stream_ctrl->ftp_ctrl->mode == PASSIVE_FTP) {	status_code = ftp_passive_mode(stream,stream->serverinfo->host,&data_sock);	if((!is_ftp_OK(status_code)) || (data_sock <= 0)) {	    stream_ctrl->ftp_ctrl->mode = ACTIVE_FTP;	}    }            /*      PORT    */    if(stream_ctrl->ftp_ctrl->mode == ACTIVE_FTP) {	status_code = ftp_tell_port(stream,&data_wait_sock);	if((!is_ftp_OK(status_code)) || (data_wait_sock <= 0)) {	    goto failed;	}    }            if(stream_ctrl->ftp_ctrl->mode == ACTIVE_FTP) {	data_sock = accept_connection(data_wait_sock);	if(data_sock < 0) {	    display(MSDL_ERR,"could not establish data connection\n");	    goto failed;	}    }            /*      TYPE    */    status_code = ftp_bin_mode(stream);        /*      REST     */    if(stream->dlopts->resume_download) { /* try resume, by setting byterange */	ftp_prepare_resuming(stream);    }    if(stream->dlopts->byterange) {	status_code = ftp_seek_pos(stream);		if(is_ftp_OK(status_code)) {	    if(stream->dlopts->resume_download) {		stream->resumeinfo->resume_req_success = 1;		display(MSDL_VER,"ftp resume OK\n");	    }	    else {		display(MSDL_VER,"ftp seek OK\n");	    }	}	else {	    display(MSDL_ERR,"ftp resume failed\n");	    if(stream->dlopts->resume_download) {		stream->resumeinfo->resume_start_offset = 0; /* starting from beginning */		stream->resumeinfo->resume_req_success = 0;		display(MSDL_ERR,"ftp: download file from beginning\n");	    }	}    }        /*      SIZE    */    status_code = ftp_ask_size(stream,url->filepath,&file_size);    if(file_size) {	stream_ctrl->file_size = stream_ctrl->ftp_ctrl->file_size = file_size;    }        /*      RETR    */    status_code = ftp_retr(stream,url->filepath);    if(!is_ftp_OK(status_code)) {	goto failed;    }                stream->stream_ctrl->protocol = FTP;        stream_ctrl->ftp_ctrl->command_sock = command_sock; /* just to make sure */    stream_ctrl->ftp_ctrl->data_wait_sock = data_wait_sock;    stream_ctrl->ftp_ctrl->data_sock = data_sock;    stream_ctrl->ftp_ctrl->down_size = 0;    stream_ctrl->status = STREAMING_DOWNLOADING;    /* stream_sock is CURRENT using socket, so now it's changed to data_sock */    stream->netsock->sock = data_sock;        return 1;  failed:    return 0;}/* * ftp stream. filles buffer, and buffer size is 'size'  * ftp is special. it read from socket directory, as we don't need * complicated buffering mechanism, such as in RTSP or so. * *  return value: bytes written to buffer. */int ftp_streaming_read(struct stream_t *stream,		       uint8_t *buffer, size_t buffer_size){    struct ftp_ctrl_t *ftp_ctrl = stream->stream_ctrl->ftp_ctrl;    size_t read_len = 0;    size_t ask_size = 0;        /*transfer_force_end_size*/    if(ftp_ctrl->transfer_force_end_size) {	if(ftp_ctrl->down_size >= ftp_ctrl->transfer_force_end_size) { /* already finished */	    stream->stream_ctrl->status = STREAMING_FINISHED;	    return 0;	}	else {	    size_t size_diff = (size_t)(ftp_ctrl->transfer_force_end_size - ftp_ctrl->down_size);	    ask_size = (size_diff > buffer_size) ? buffer_size : size_diff; /* min */	}    }    else {	ask_size = buffer_size;    }        read_len = xrecv(ftp_ctrl->data_sock,buffer,ask_size);    if(read_len < 0) {	perror("recv() error");	return -1;    }    if(read_len == 0) {	stream->stream_ctrl->status = STREAMING_FINISHED;    }        ftp_ctrl->down_size += read_len;    return read_len;}struct stream_t *ftp_streaming_init(){    struct stream_t *stream = streaming_init_common();    stream->stream_ctrl->ftp_ctrl = new_ftp_ctrl_t();      stream->start = ftp_streaming_start;    stream->read  = ftp_streaming_read;    stream->close = ftp_streaming_close;    return stream;}void ftp_streaming_close(struct stream_t *stream){    /*      FTP uses 2(or 3 in active) sockets, and they are closed in free_ftp_ctrl_t    */    ftp_send_command(stream,"QUIT\n"); /* to be nice to server. */        free_ftp_ctrl_t(stream->stream_ctrl->ftp_ctrl);    streaming_close_common(stream);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -