2017-01-28 04:49:09 +08:00
# include "pch.h"
2016-09-19 11:50:08 +08:00
# include "vcpkg_cmd_arguments.h"
# include "vcpkg_Commands.h"
# include "metrics.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 " ) ;
2017-01-13 14:03:57 +08:00
Commands : : Help : : print_usage ( ) ;
2016-09-19 11:50:08 +08:00
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 " ) ;
2017-01-13 14:03:57 +08:00
Commands : : Help : : print_usage ( ) ;
2016-09-19 11:50:08 +08:00
exit ( EXIT_FAILURE ) ;
}
option_field = std : : make_unique < std : : string > ( * arg_begin ) ;
}
static void parse_switch (
2017-02-01 05:31:45 +08:00
opt_bool_t new_setting ,
2016-09-19 11:50:08 +08:00
const std : : string & option_name ,
2017-02-01 05:31:45 +08:00
opt_bool_t & option_field )
2016-09-19 11:50:08 +08:00
{
2017-02-01 05:31:45 +08:00
if ( option_field ! = opt_bool_t : : UNSPECIFIED & & option_field ! = new_setting )
2016-09-19 11:50:08 +08:00
{
System : : println ( System : : color : : error , " Error: conflicting values specified for --%s " , option_name ) ;
TrackProperty ( " error " , " error conflicting switches " ) ;
2017-01-13 14:03:57 +08:00
Commands : : Help : : print_usage ( ) ;
2016-09-19 11:50:08 +08:00
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 ] ! = ' - ' )
{
TrackProperty ( " error " , " error short options are not supported " ) ;
2017-03-23 08:06:09 +08:00
Checks : : exit_with_message ( VCPKG_LINE_INFO , " Error: short options are not supported: %s " , arg ) ;
2016-09-19 11:50:08 +08:00
}
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 " )
{
2017-02-01 05:31:45 +08:00
parse_switch ( opt_bool_t : : ENABLED , " debug " , args . debug ) ;
2016-09-19 11:50:08 +08:00
continue ;
}
if ( arg = = " --sendmetrics " )
{
2017-02-01 05:31:45 +08:00
parse_switch ( opt_bool_t : : ENABLED , " sendmetrics " , args . sendmetrics ) ;
2016-09-19 11:50:08 +08:00
continue ;
}
if ( arg = = " --printmetrics " )
{
2017-02-01 05:31:45 +08:00
parse_switch ( opt_bool_t : : ENABLED , " printmetrics " , args . printmetrics ) ;
2016-09-19 11:50:08 +08:00
continue ;
}
if ( arg = = " --no-sendmetrics " )
{
2017-02-01 05:31:45 +08:00
parse_switch ( opt_bool_t : : DISABLED , " sendmetrics " , args . sendmetrics ) ;
2016-09-19 11:50:08 +08:00
continue ;
}
if ( arg = = " --no-printmetrics " )
{
2017-02-01 05:31:45 +08:00
parse_switch ( opt_bool_t : : DISABLED , " printmetrics " , args . printmetrics ) ;
2016-09-19 11:50:08 +08:00
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 )
{
2016-12-13 07:10:29 +08:00
System : : println ( option ) ;
2016-09-19 11:50:08 +08:00
}
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 , " " ) ;
}
2016-12-13 07:03:36 +08:00
void vcpkg_cmd_arguments : : check_max_arg_count ( const size_t expected_arg_count , const std : : string & example_text ) const
2016-10-01 02:24:04 +08:00
{
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 ) ;
2016-10-01 02:24:04 +08:00
System : : print ( example_text ) ;
exit ( EXIT_FAILURE ) ;
}
}
2016-12-13 07:03:36 +08:00
void vcpkg_cmd_arguments : : check_min_arg_count ( const size_t expected_arg_count , const std : : string & example_text ) const
2016-10-01 02:24:04 +08:00
{
const size_t actual_arg_count = command_arguments . size ( ) ;
if ( actual_arg_count < expected_arg_count )
{
2016-10-01 02:24:04 +08:00
System : : println ( System : : color : : error , " Error: `%s` requires at least %u arguments, but %u were provided " , this - > command , expected_arg_count , actual_arg_count ) ;
2016-10-01 02:24:04 +08:00
System : : print ( example_text ) ;
exit ( EXIT_FAILURE ) ;
}
}
2016-12-13 07:03:36 +08:00
void vcpkg_cmd_arguments : : check_exact_arg_count ( const size_t expected_arg_count , const std : : string & example_text ) const
2016-10-01 02:24:04 +08:00
{
const size_t actual_arg_count = command_arguments . size ( ) ;
if ( actual_arg_count ! = expected_arg_count )
{
2016-10-01 02:24:04 +08:00
System : : println ( System : : color : : error , " Error: `%s` requires %u arguments, but %u were provided " , this - > command , expected_arg_count , actual_arg_count ) ;
2016-10-01 02:24:04 +08:00
System : : print ( example_text ) ;
2016-09-19 11:50:08 +08:00
exit ( EXIT_FAILURE ) ;
}
}
}