123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241 |
- #ifndef BOOST_UUID_STRING_GENERATOR_HPP_INCLUDED
- #define BOOST_UUID_STRING_GENERATOR_HPP_INCLUDED
- // Copyright 2010 Andy Tompkins
- // Copyright 2024 Peter Dimov
- // Distributed under the Boost Software License, Version 1.0.
- // https://www.boost.org/LICENSE_1_0.txt
- #include <boost/uuid/uuid.hpp>
- #include <boost/throw_exception.hpp>
- #include <boost/config.hpp>
- #include <string>
- #include <iterator>
- #include <algorithm> // for find
- #include <stdexcept>
- #include <cstring> // for strlen, wcslen
- #include <cstdio>
- namespace boost {
- namespace uuids {
- // Generates a UUID from a string
- //
- // Accepts the following forms:
- //
- // 0123456789abcdef0123456789abcdef
- // 01234567-89ab-cdef-0123-456789abcdef
- // {01234567-89ab-cdef-0123-456789abcdef}
- // {0123456789abcdef0123456789abcdef}
- struct string_generator
- {
- using result_type = uuid;
- template<class Ch, class Traits, class Alloc>
- uuid operator()( std::basic_string<Ch, Traits, Alloc> const& s ) const
- {
- return operator()(s.begin(), s.end());
- }
- uuid operator()( char const* s ) const
- {
- return operator()( s, s + std::strlen( s ) );
- }
- uuid operator()( wchar_t const* s ) const
- {
- return operator()( s, s + std::wcslen( s ) );
- }
- template<class CharIterator>
- uuid operator()( CharIterator begin, CharIterator end ) const
- {
- using char_type = typename std::iterator_traits<CharIterator>::value_type;
- int ipos = 0;
- // check open brace
- char_type c = get_next_char( begin, end, ipos );
- bool has_open_brace = is_open_brace( c );
- char_type open_brace_char = c;
- if( has_open_brace )
- {
- c = get_next_char( begin, end, ipos );
- }
- bool has_dashes = false;
- uuid u;
- int i = 0;
- for( uuid::iterator it_byte = u.begin(); it_byte != u.end(); ++it_byte, ++i )
- {
- if( it_byte != u.begin() )
- {
- c = get_next_char( begin, end, ipos );
- }
- if( i == 4 )
- {
- has_dashes = is_dash( c );
- if( has_dashes )
- {
- c = get_next_char( begin, end, ipos );
- }
- }
- else if( i == 6 || i == 8 || i == 10 )
- {
- // if there are dashes, they must be in every slot
- if( has_dashes )
- {
- if( is_dash( c ) )
- {
- c = get_next_char( begin, end, ipos );
- }
- else
- {
- throw_invalid( ipos - 1, "dash expected" );
- }
- }
- }
- *it_byte = get_value( c, ipos - 1 );
- c = get_next_char( begin, end, ipos );
- *it_byte <<= 4;
- *it_byte |= get_value( c, ipos - 1 );
- }
- // check close brace
- if( has_open_brace )
- {
- c = get_next_char( begin, end, ipos );
- check_close_brace( c, open_brace_char, ipos - 1 );
- }
- // check end of string - any additional data is an invalid uuid
- if( begin != end )
- {
- throw_invalid( ipos, "unexpected extra input" );
- }
- return u;
- }
- private:
- BOOST_NORETURN void throw_invalid( int ipos, char const* error ) const
- {
- char buffer[ 16 ];
- std::snprintf( buffer, sizeof( buffer ), "%d", ipos );
- BOOST_THROW_EXCEPTION( std::runtime_error( std::string( "Invalid UUID string at position " ) + buffer + ": " + error ) );
- }
- template <typename CharIterator>
- typename std::iterator_traits<CharIterator>::value_type
- get_next_char( CharIterator& begin, CharIterator end, int& ipos ) const
- {
- if( begin == end )
- {
- throw_invalid( ipos, "unexpected end of input" );
- }
- ++ipos;
- return *begin++;
- }
- unsigned char get_value( char c, int ipos ) const
- {
- static char const digits_begin[] = "0123456789abcdefABCDEF";
- static size_t digits_len = (sizeof(digits_begin) / sizeof(char)) - 1;
- static char const* const digits_end = digits_begin + digits_len;
- static unsigned char const values[] =
- { 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,10,11,12,13,14,15 };
- size_t pos = std::find( digits_begin, digits_end, c ) - digits_begin;
- if( pos >= digits_len )
- {
- throw_invalid( ipos, "hex digit expected" );
- }
- return values[ pos ];
- }
- unsigned char get_value( wchar_t c, int ipos ) const
- {
- static wchar_t const digits_begin[] = L"0123456789abcdefABCDEF";
- static size_t digits_len = (sizeof(digits_begin) / sizeof(wchar_t)) - 1;
- static wchar_t const* const digits_end = digits_begin + digits_len;
- static unsigned char const values[] =
- { 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,10,11,12,13,14,15 };
- size_t pos = std::find( digits_begin, digits_end, c ) - digits_begin;
- if( pos >= digits_len )
- {
- throw_invalid( ipos, "hex digit expected" );
- }
- return values[ pos ];
- }
- bool is_dash( char c ) const
- {
- return c == '-';
- }
- bool is_dash( wchar_t c ) const
- {
- return c == L'-';
- }
- // return closing brace
- bool is_open_brace( char c ) const
- {
- return c == '{';
- }
- bool is_open_brace( wchar_t c ) const
- {
- return c == L'{';
- }
- void check_close_brace( char c, char open_brace, int ipos ) const
- {
- if( open_brace == '{' && c == '}' )
- {
- //great
- }
- else
- {
- throw_invalid( ipos, "closing brace expected" );
- }
- }
- void check_close_brace( wchar_t c, wchar_t open_brace, int ipos ) const
- {
- if( open_brace == L'{' && c == L'}' )
- {
- // great
- }
- else
- {
- throw_invalid( ipos, "closing brace expected" );
- }
- }
- };
- }} // namespace boost::uuids
- #endif // BOOST_UUID_STRING_GENERATOR_HPP_INCLUDED
|