2016-09-19 11:50:08 +08:00
# define WIN32_LEAN_AND_MEAN
# include <Windows.h>
# include "vcpkg_cmd_arguments.h"
# include "vcpkg_Commands.h"
# include "vcpkg_Graphs.h"
# include <unordered_set>
# include "metrics.h"
# include "vcpkg.h"
# include "vcpkg_System.h"
namespace vcpkg
{
static void parse_value (
const std : : string * arg_begin ,
const std : : string * arg_end ,
const std : : string & option_name ,
std : : unique_ptr < std : : string > & option_field )
{
if ( arg_begin = = arg_end )
{
System : : println ( System : : color : : error , " Error: expected value after %s " , option_name ) ;
TrackProperty ( " error " , " error option name " ) ;
print_usage ( ) ;
exit ( EXIT_FAILURE ) ;
}
if ( option_field ! = nullptr )
{
System : : println ( System : : color : : error , " Error: %s specified multiple times " , option_name ) ;
TrackProperty ( " error " , " error option specified multiple times " ) ;
print_usage ( ) ;
exit ( EXIT_FAILURE ) ;
}
option_field = std : : make_unique < std : : string > ( * arg_begin ) ;
}
static void parse_switch (
opt_bool new_setting ,
const std : : string & option_name ,
opt_bool & option_field )
{
if ( option_field ! = opt_bool : : unspecified & & option_field ! = new_setting )
{
System : : println ( System : : color : : error , " Error: conflicting values specified for --%s " , option_name ) ;
TrackProperty ( " error " , " error conflicting switches " ) ;
print_usage ( ) ;
exit ( EXIT_FAILURE ) ;
}
option_field = new_setting ;
}
vcpkg_cmd_arguments vcpkg_cmd_arguments : : create_from_command_line ( const int argc , const wchar_t * const * const argv )
{
std : : vector < std : : string > v ;
for ( int i = 1 ; i < argc ; + + i )
{
v . push_back ( Strings : : utf16_to_utf8 ( argv [ i ] ) ) ;
}
return vcpkg_cmd_arguments : : create_from_arg_sequence ( v . data ( ) , v . data ( ) + v . size ( ) ) ;
}
vcpkg_cmd_arguments vcpkg_cmd_arguments : : create_from_arg_sequence ( const std : : string * arg_begin , const std : : string * arg_end )
{
vcpkg_cmd_arguments args ;
for ( ; arg_begin ! = arg_end ; + + arg_begin )
{
std : : string arg = * arg_begin ;
if ( arg . empty ( ) )
{
continue ;
}
if ( arg [ 0 ] = = ' - ' & & arg [ 1 ] ! = ' - ' )
{
System : : println ( System : : color : : error , " Error: short options are not supported: %s " , arg ) ;
TrackProperty ( " error " , " error short options are not supported " ) ;
exit ( EXIT_FAILURE ) ;
}
if ( arg [ 0 ] = = ' - ' & & arg [ 1 ] = = ' - ' )
{
// command switch
if ( arg = = " --vcpkg-root " )
{
+ + arg_begin ;
parse_value ( arg_begin , arg_end , " --vcpkg-root " , args . vcpkg_root_dir ) ;
continue ;
}
if ( arg = = " --triplet " )
{
+ + arg_begin ;
parse_value ( arg_begin , arg_end , " --triplet " , args . target_triplet ) ;
continue ;
}
if ( arg = = " --debug " )
{
parse_switch ( opt_bool : : enabled , " debug " , args . debug ) ;
continue ;
}
if ( arg = = " --sendmetrics " )
{
parse_switch ( opt_bool : : enabled , " sendmetrics " , args . sendmetrics ) ;
continue ;
}
if ( arg = = " --printmetrics " )
{
parse_switch ( opt_bool : : enabled , " printmetrics " , args . printmetrics ) ;
continue ;
}
if ( arg = = " --no-sendmetrics " )
{
parse_switch ( opt_bool : : disabled , " sendmetrics " , args . sendmetrics ) ;
continue ;
}
if ( arg = = " --no-printmetrics " )
{
parse_switch ( opt_bool : : disabled , " printmetrics " , args . printmetrics ) ;
continue ;
}
args . optional_command_arguments . insert ( arg ) ;
continue ;
}
if ( args . command . empty ( ) )
{
args . command = arg ;
}
else
{
args . command_arguments . push_back ( arg ) ;
}
}
return args ;
}
std : : unordered_set < std : : string > vcpkg_cmd_arguments : : check_and_get_optional_command_arguments ( const std : : vector < std : : string > & valid_options ) const
{
std : : unordered_set < std : : string > output ;
auto options_copy = this - > optional_command_arguments ;
for ( const std : : string & option : valid_options )
{
auto it = options_copy . find ( option ) ;
if ( it ! = options_copy . end ( ) )
{
output . insert ( option ) ;
options_copy . erase ( it ) ;
}
}
if ( ! options_copy . empty ( ) )
{
System : : println ( System : : color : : error , " Unknown option(s) for command '%s': " , this - > command ) ;
for ( const std : : string & option : options_copy )
{
System : : println ( option . c_str ( ) ) ;
}
exit ( EXIT_FAILURE ) ;
}
return output ;
}
2016-10-01 02:24:04 +08:00
void vcpkg_cmd_arguments : : check_max_arg_count ( const size_t expected_arg_count ) const
2016-09-19 11:50:08 +08:00
{
2016-10-01 02:24:04 +08:00
return check_max_arg_count ( expected_arg_count , " " ) ;
}
void vcpkg_cmd_arguments : : check_min_arg_count ( const size_t expected_arg_count ) const
{
return check_min_arg_count ( expected_arg_count , " " ) ;
}
void vcpkg_cmd_arguments : : check_exact_arg_count ( const size_t expected_arg_count ) const
{
return check_exact_arg_count ( expected_arg_count , " " ) ;
}
void vcpkg_cmd_arguments : : check_max_arg_count ( const size_t expected_arg_count , const char * example_text ) const
{
const size_t actual_arg_count = command_arguments . size ( ) ;
if ( actual_arg_count > expected_arg_count )
2016-09-19 11:50:08 +08:00
{
2016-10-01 02:24:04 +08:00
System : : println ( System : : color : : error , " Error: %s requires at most %u arguments, but %u were provided " , this - > command , expected_arg_count , actual_arg_count ) ;
System : : print ( example_text ) ;
exit ( EXIT_FAILURE ) ;
}
}
void vcpkg_cmd_arguments : : check_min_arg_count ( const size_t expected_arg_count , const char * example_text ) const
{
const size_t actual_arg_count = command_arguments . size ( ) ;
if ( actual_arg_count < expected_arg_count )
{
System : : println ( System : : color : : error , " Error: %s requires at least %u arguments, but %u were provided " , this - > command , expected_arg_count , actual_arg_count ) ;
System : : print ( example_text ) ;
exit ( EXIT_FAILURE ) ;
}
}
void vcpkg_cmd_arguments : : check_exact_arg_count ( const size_t expected_arg_count , const char * example_text ) const
{
const size_t actual_arg_count = command_arguments . size ( ) ;
if ( actual_arg_count ! = expected_arg_count )
{
System : : println ( System : : color : : error , " Error: %s requires %u arguments, but %u were provided " , this - > command , expected_arg_count , actual_arg_count ) ;
System : : print ( example_text ) ;
2016-09-19 11:50:08 +08:00
exit ( EXIT_FAILURE ) ;
}
}
2016-09-24 07:26:27 +08:00
std : : vector < package_spec > vcpkg_cmd_arguments : : parse_all_arguments_as_package_specs ( const triplet & default_target_triplet , const char * example_text ) const
2016-09-19 11:50:08 +08:00
{
size_t arg_count = command_arguments . size ( ) ;
if ( arg_count < 1 )
{
System : : println ( System : : color : : error , " Error: %s requires one or more package specifiers " , this - > command ) ;
if ( example_text = = nullptr )
print_example ( Strings : : format ( " %s zlib zlib:x64-windows curl boost " , this - > command ) . c_str ( ) ) ;
else
print_example ( example_text ) ;
exit ( EXIT_FAILURE ) ;
}
std : : vector < package_spec > specs ;
specs . reserve ( arg_count ) ;
for ( const std : : string & command_argument : command_arguments )
{
2016-09-24 07:39:07 +08:00
expected < package_spec > current_spec = package_spec : : from_string ( command_argument , default_target_triplet ) ;
2016-09-19 11:50:08 +08:00
if ( auto spec = current_spec . get ( ) )
{
specs . push_back ( std : : move ( * spec ) ) ;
}
else
{
System : : println ( System : : color : : error , " Error: %s: %s " , current_spec . error_code ( ) . message ( ) , command_argument ) ;
print_example ( Strings : : format ( " %s zlib:x64-windows " , this - > command ) . c_str ( ) ) ;
exit ( EXIT_FAILURE ) ;
}
}
return specs ;
}
}