2020-10-11 13:57:43 -04:00
/*
* Copyright ( c ) 2020 , the SerenityOS developers .
2023-06-17 09:56:59 -04:00
* Copyright ( c ) 2023 , Sam Atkins < atkinssj @ serenityos . org >
2020-10-11 13:57:43 -04:00
*
2021-04-22 04:24:48 -04:00
* SPDX - License - Identifier : BSD - 2 - Clause
2020-10-11 13:57:43 -04:00
*/
# pragma once
2022-12-04 13:02:33 -05:00
# include <AK/DeprecatedString.h>
2020-10-11 13:57:43 -04:00
# include <AK/GenericLexer.h>
# include <AK/HashMap.h>
2023-06-17 09:56:59 -04:00
# include <AK/String.h>
2020-10-11 13:57:43 -04:00
# include <AK/StringBuilder.h>
namespace AK {
class SourceGenerator {
2020-10-23 12:36:56 -04:00
AK_MAKE_NONCOPYABLE ( SourceGenerator ) ;
2020-10-11 13:57:43 -04:00
public :
2023-06-17 09:56:59 -04:00
using MappingType = HashMap < StringView , String > ;
2020-10-11 13:57:43 -04:00
2020-10-23 12:36:56 -04:00
explicit SourceGenerator ( StringBuilder & builder , char opening = ' @ ' , char closing = ' @ ' )
: m_builder ( builder )
2020-10-11 13:57:43 -04:00
, m_opening ( opening )
, m_closing ( closing )
{
}
2022-04-01 13:58:27 -04:00
explicit SourceGenerator ( StringBuilder & builder , MappingType const & mapping , char opening = ' @ ' , char closing = ' @ ' )
2020-10-23 12:36:56 -04:00
: m_builder ( builder )
2023-05-13 14:26:20 -04:00
, m_mapping ( mapping . clone ( ) . release_value_but_fixme_should_propagate_errors ( ) )
2020-10-11 13:57:43 -04:00
, m_opening ( opening )
, m_closing ( closing )
{
}
2022-02-22 13:20:08 -05:00
SourceGenerator ( SourceGenerator & & ) = default ;
2020-10-23 12:36:56 -04:00
SourceGenerator fork ( ) { return SourceGenerator { m_builder , m_mapping , m_opening , m_closing } ; }
2020-10-11 13:57:43 -04:00
2023-06-17 09:56:59 -04:00
ErrorOr < void > set ( StringView key , String value )
2022-08-23 11:40:39 -04:00
{
if ( key . contains ( m_opening ) | | key . contains ( m_closing ) ) {
warnln ( " SourceGenerator keys cannot contain the opening/closing delimiters `{}` and `{}`. (Keys are only wrapped in these when using them, not when setting them.) " , m_opening , m_closing ) ;
VERIFY_NOT_REACHED ( ) ;
}
2023-06-17 09:56:59 -04:00
TRY ( m_mapping . try_set ( key , move ( value ) ) ) ;
return { } ;
}
String get ( StringView key ) const
2022-03-08 12:29:45 -05:00
{
auto result = m_mapping . get ( key ) ;
if ( ! result . has_value ( ) ) {
warnln ( " No key named `{}` set on SourceGenerator " , key ) ;
VERIFY_NOT_REACHED ( ) ;
}
return result . release_value ( ) ;
}
2020-10-11 13:57:43 -04:00
2020-10-23 12:36:56 -04:00
StringView as_string_view ( ) const { return m_builder . string_view ( ) ; }
2020-10-11 13:57:43 -04:00
2023-06-17 09:56:03 -04:00
ErrorOr < void > try_append ( StringView pattern )
2020-10-11 13:57:43 -04:00
{
GenericLexer lexer { pattern } ;
while ( ! lexer . is_eof ( ) ) {
2023-06-17 09:56:03 -04:00
TRY ( m_builder . try_append ( lexer . consume_until ( m_opening ) ) ) ;
2020-10-11 13:57:43 -04:00
if ( lexer . consume_specific ( m_opening ) ) {
2023-06-16 12:23:59 -04:00
auto const placeholder = lexer . consume_until ( m_closing ) ;
2020-10-11 13:57:43 -04:00
if ( ! lexer . consume_specific ( m_closing ) )
2021-02-23 14:42:32 -05:00
VERIFY_NOT_REACHED ( ) ;
2020-10-11 13:57:43 -04:00
2023-06-17 09:56:03 -04:00
TRY ( m_builder . try_append ( get ( placeholder ) ) ) ;
2020-10-11 13:57:43 -04:00
} else {
2021-02-23 14:42:32 -05:00
VERIFY ( lexer . is_eof ( ) ) ;
2020-10-11 13:57:43 -04:00
}
}
2023-06-17 09:56:03 -04:00
return { } ;
2020-10-11 13:57:43 -04:00
}
2023-06-17 09:56:03 -04:00
ErrorOr < void > try_appendln ( StringView pattern )
2022-02-22 13:21:33 -05:00
{
2023-06-17 09:56:03 -04:00
TRY ( try_append ( pattern ) ) ;
TRY ( m_builder . try_append ( ' \n ' ) ) ;
return { } ;
2022-02-22 13:21:33 -05:00
}
2022-07-11 20:58:29 -04:00
template < size_t N >
2023-06-17 09:56:59 -04:00
String get ( char const ( & key ) [ N ] )
2022-07-11 20:58:29 -04:00
{
return get ( StringView { key , N - 1 } ) ;
}
template < size_t N >
2023-06-17 09:56:03 -04:00
ErrorOr < void > set ( char const ( & key ) [ N ] , String value )
2022-07-11 20:58:29 -04:00
{
2023-06-17 09:56:03 -04:00
return set ( StringView { key , N - 1 } , value ) ;
2022-07-11 20:58:29 -04:00
}
2023-06-17 09:56:59 -04:00
template < size_t N >
2023-06-17 09:56:03 -04:00
ErrorOr < void > try_append ( char const ( & pattern ) [ N ] )
2023-06-17 09:56:59 -04:00
{
2023-06-17 09:56:03 -04:00
return try_append ( StringView { pattern , N - 1 } ) ;
2023-06-17 09:56:59 -04:00
}
2022-07-11 20:58:29 -04:00
template < size_t N >
2023-06-17 09:56:03 -04:00
ErrorOr < void > try_appendln ( char const ( & pattern ) [ N ] )
2022-07-11 20:58:29 -04:00
{
2023-06-17 09:56:03 -04:00
return try_appendln ( StringView { pattern , N - 1 } ) ;
2022-07-11 20:58:29 -04:00
}
2023-06-17 09:56:03 -04:00
// FIXME: These are deprecated.
void set ( StringView key , DeprecatedString value )
{
MUST ( set ( key , MUST ( String : : from_deprecated_string ( value ) ) ) ) ;
}
2022-07-11 20:58:29 -04:00
template < size_t N >
2023-06-17 09:56:03 -04:00
void set ( char const ( & key ) [ N ] , DeprecatedString value )
2022-07-11 20:58:29 -04:00
{
2023-06-17 09:56:03 -04:00
set ( StringView { key , N - 1 } , value ) ;
2022-07-11 20:58:29 -04:00
}
2023-06-17 09:56:03 -04:00
void append ( StringView pattern ) { MUST ( try_append ( pattern ) ) ; }
void appendln ( StringView pattern ) { MUST ( try_appendln ( pattern ) ) ; }
template < size_t N >
void append ( char const ( & pattern ) [ N ] ) { MUST ( try_append ( pattern ) ) ; }
template < size_t N >
void appendln ( char const ( & pattern ) [ N ] ) { MUST ( try_appendln ( pattern ) ) ; }
2022-07-11 20:58:29 -04:00
2020-10-11 13:57:43 -04:00
private :
2020-10-23 12:36:56 -04:00
StringBuilder & m_builder ;
2020-10-11 13:57:43 -04:00
MappingType m_mapping ;
char m_opening , m_closing ;
} ;
}
2022-11-26 06:18:30 -05:00
# if USING_AK_GLOBALLY
2020-10-11 13:57:43 -04:00
using AK : : SourceGenerator ;
2022-11-26 06:18:30 -05:00
# endif