📄 ftp.c
字号:
/***************************************************************************** * ftp.c: FTP input module ***************************************************************************** * Copyright (C) 2001-2006 the VideoLAN team * $Id: ftp.c 16774 2006-09-21 19:29:10Z hartman $ * * Authors: Laurent Aimar <fenrir@via.ecp.fr> - original code * Rémi Denis-Courmont <rem # videolan.org> - EPSV support * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#include <stdlib.h>#include <vlc/vlc.h>#include <vlc/input.h>#include "network.h"#include "vlc_url.h"/***************************************************************************** * Module descriptor *****************************************************************************/static int Open ( vlc_object_t * );static void Close( vlc_object_t * );#define CACHING_TEXT N_("Caching value in ms")#define CACHING_LONGTEXT N_( \ "Caching value for FTP streams. This " \ "value should be set in milliseconds." )#define USER_TEXT N_("FTP user name")#define USER_LONGTEXT N_("User name that will " \ "be used for the connection.")#define PASS_TEXT N_("FTP password")#define PASS_LONGTEXT N_("Password that will be " \ "used for the connection.")#define ACCOUNT_TEXT N_("FTP account")#define ACCOUNT_LONGTEXT N_("Account that will be " \ "used for the connection.")vlc_module_begin(); set_shortname( "FTP" ); set_description( _("FTP input") ); set_capability( "access2", 0 ); set_category( CAT_INPUT ); set_subcategory( SUBCAT_INPUT_ACCESS ); add_integer( "ftp-caching", 2 * DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE ); add_string( "ftp-user", "anonymous", NULL, USER_TEXT, USER_LONGTEXT, VLC_FALSE ); add_string( "ftp-pwd", "anonymous@example.com", NULL, PASS_TEXT, PASS_LONGTEXT, VLC_FALSE ); add_string( "ftp-account", "anonymous", NULL, ACCOUNT_TEXT, ACCOUNT_LONGTEXT, VLC_FALSE ); add_shortcut( "ftp" ); set_callbacks( Open, Close );vlc_module_end();/***************************************************************************** * Local prototypes *****************************************************************************/static int Read( access_t *, uint8_t *, int );static int Seek( access_t *, int64_t );static int Control( access_t *, int, va_list );struct access_sys_t{ vlc_url_t url; int fd_cmd; int fd_data; char sz_epsv_ip[NI_MAXNUMERICHOST];};static int ftp_SendCommand( access_t *, char *, ... );static int ftp_ReadCommand( access_t *, int *, char ** );static int ftp_StartStream( access_t *, int64_t );static int ftp_StopStream ( access_t *);static int Connect( access_t *p_access, access_sys_t *p_sys ){ int fd, i_answer; char *psz; /* *** Open a TCP connection with server *** */ msg_Dbg( p_access, "waiting for connection..." ); p_sys->fd_cmd = fd = net_ConnectTCP( p_access, p_sys->url.psz_host, p_sys->url.i_port ); if( fd < 0 ) { msg_Err( p_access, "failed to connect with server" ); return -1; } for( ;; ) { if( ftp_ReadCommand( p_access, &i_answer, NULL ) != 1 ) { break; } } if( i_answer / 100 != 2 ) { msg_Err( p_access, "connection rejected" ); return -1; } msg_Dbg( p_access, "connection accepted (%d)", i_answer ); if( p_sys->url.psz_username && *p_sys->url.psz_username ) psz = strdup( p_sys->url.psz_username ); else psz = var_CreateGetString( p_access, "ftp-user" ); if( ftp_SendCommand( p_access, "USER %s", psz ) < 0 || ftp_ReadCommand( p_access, &i_answer, NULL ) < 0 ) { free( psz ); return -1; } free( psz ); switch( i_answer / 100 ) { case 2: msg_Dbg( p_access, "user accepted" ); break; case 3: msg_Dbg( p_access, "password needed" ); if( p_sys->url.psz_password && *p_sys->url.psz_password ) psz = strdup( p_sys->url.psz_password ); else psz = var_CreateGetString( p_access, "ftp-pwd" ); if( ftp_SendCommand( p_access, "PASS %s", psz ) < 0 || ftp_ReadCommand( p_access, &i_answer, NULL ) < 0 ) { free( psz ); return -1; } free( psz ); switch( i_answer / 100 ) { case 2: msg_Dbg( p_access, "password accepted" ); break; case 3: msg_Dbg( p_access, "account needed" ); psz = var_CreateGetString( p_access, "ftp-account" ); if( ftp_SendCommand( p_access, "ACCT %s", psz ) < 0 || ftp_ReadCommand( p_access, &i_answer, NULL ) < 0 ) { free( psz ); return -1; } free( psz ); if( i_answer / 100 != 2 ) { msg_Err( p_access, "account rejected" ); return -1; } msg_Dbg( p_access, "account accepted" ); break; default: msg_Err( p_access, "password rejected" ); return -1; } break; default: msg_Err( p_access, "user rejected" ); return -1; } return 0;}/**************************************************************************** * Open: connect to ftp server and ask for file ****************************************************************************/static int Open( vlc_object_t *p_this ){ access_t *p_access = (access_t*)p_this; access_sys_t *p_sys; char *psz; int i_answer; char *psz_arg; /* Init p_access */ p_access->pf_read = Read; p_access->pf_block = NULL; p_access->pf_seek = Seek; p_access->pf_control = Control; p_access->info.i_update = 0; p_access->info.i_size = 0; p_access->info.i_pos = 0; p_access->info.b_eof = VLC_FALSE; p_access->info.i_title = 0; p_access->info.i_seekpoint = 0; p_access->p_sys = p_sys = malloc( sizeof( access_sys_t ) ); memset( p_sys, 0, sizeof( access_sys_t ) ); p_sys->fd_cmd = -1; p_sys->fd_data = -1; /* *** Parse URL and get server addr/port and path *** */ psz = p_access->psz_path; while( *psz == '/' ) { psz++; } vlc_UrlParse( &p_sys->url, psz, 0 ); if( p_sys->url.psz_host == NULL || *p_sys->url.psz_host == '\0' ) { msg_Err( p_access, "invalid server name" ); goto exit_error; } if( p_sys->url.i_port <= 0 ) { p_sys->url.i_port = 21; /* default port */ } /* FTP URLs are relative to user's default directory (RFC1738) For absolute path use ftp://foo.bar//usr/local/etc/filename */ if( *p_sys->url.psz_path == '/' ) p_sys->url.psz_path++; if( Connect( p_access, p_sys ) < 0 ) goto exit_error; /* Extended passive mode */ if( ftp_SendCommand( p_access, "EPSV ALL" ) < 0 ) { msg_Err( p_access, "cannot request extended passive mode" ); return -1; } if( ftp_ReadCommand( p_access, &i_answer, NULL ) == 2 ) { if( net_GetPeerAddress( p_sys->fd_cmd, p_sys->sz_epsv_ip, NULL ) ) goto exit_error; } else { /* If ESPV ALL fails, we fallback to PASV. * We have to restart the connection in case there is a NAT that * understands EPSV ALL in the way, and hence won't allow PASV on * the initial connection. */ net_Close( p_sys->fd_cmd ); p_sys->fd_cmd = -1; *p_sys->sz_epsv_ip = '\0'; msg_Info( p_access, "FTP Extended passive mode disabled" ); if( ( p_sys->fd_cmd = Connect( p_access, p_sys ) ) < 0 ) goto exit_error; } /* binary mode */ if( ftp_SendCommand( p_access, "TYPE I" ) < 0 || ftp_ReadCommand( p_access, &i_answer, NULL ) != 2 ) { msg_Err( p_access, "cannot set binary transfer mode" ); goto exit_error; } /* get size */ if( ftp_SendCommand( p_access, "SIZE %s", p_sys->url.psz_path ) < 0 || ftp_ReadCommand( p_access, &i_answer, &psz_arg ) != 2 ) { msg_Err( p_access, "cannot get file size" ); goto exit_error; } p_access->info.i_size = atoll( &psz_arg[4] ); free( psz_arg ); msg_Dbg( p_access, "file size: "I64Fd, p_access->info.i_size ); /* Start the 'stream' */ if( ftp_StartStream( p_access, 0 ) < 0 ) { msg_Err( p_access, "cannot retrieve file" ); goto exit_error; } /* Update default_pts to a suitable value for ftp access */ var_Create( p_access, "ftp-caching", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT ); return VLC_SUCCESS;exit_error: if( p_sys->fd_cmd >= 0 ) net_Close( p_sys->fd_cmd ); vlc_UrlClean( &p_sys->url ); free( p_sys ); return VLC_EGENERIC;}/***************************************************************************** * Close: free unused data structures *****************************************************************************/static void Close( vlc_object_t *p_this ){ access_t *p_access = (access_t*)p_this; access_sys_t *p_sys = p_access->p_sys; msg_Dbg( p_access, "stopping stream" ); ftp_StopStream( p_access ); if( ftp_SendCommand( p_access, "QUIT" ) < 0 ) { msg_Warn( p_access, "cannot quit" );
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -