2020-10-11 19:57:43 +02:00
/*
* Copyright ( c ) 2020 , the SerenityOS developers .
2023-06-17 14:56:59 +01:00
* Copyright ( c ) 2023 , Sam Atkins < atkinssj @ serenityos . org >
2024-07-07 19:54:59 +01:00
* Copyright ( c ) 2024 , Tim Ledbetter < timledbetter @ gmail . com >
2020-10-11 19:57:43 +02:00
*
2021-04-22 01:24:48 -07:00
* SPDX - License - Identifier : BSD - 2 - Clause
2020-10-11 19:57:43 +02:00
*/
# pragma once
2023-12-16 17:49:34 +03:30
# include <AK/ByteString.h>
2020-10-11 19:57:43 +02:00
# include <AK/GenericLexer.h>
# include <AK/HashMap.h>
2023-06-17 14:56:59 +01:00
# include <AK/String.h>
2020-10-11 19:57:43 +02:00
# include <AK/StringBuilder.h>
namespace AK {
class SourceGenerator {
2020-10-23 18:36:56 +02:00
AK_MAKE_NONCOPYABLE ( SourceGenerator ) ;
2020-10-11 19:57:43 +02:00
public :
2023-06-17 14:56:59 +01:00
using MappingType = HashMap < StringView , String > ;
2020-10-11 19:57:43 +02:00
2024-07-07 19:54:59 +01:00
explicit SourceGenerator ( StringBuilder & builder , char opening = ' @ ' , char closing = ' @ ' , char escape = ' \\ ' )
2020-10-23 18:36:56 +02:00
: m_builder ( builder )
2020-10-11 19:57:43 +02:00
, m_opening ( opening )
, m_closing ( closing )
2024-07-07 19:54:59 +01:00
, m_escape ( escape )
2020-10-11 19:57:43 +02:00
{
}
2024-07-07 19:54:59 +01:00
explicit SourceGenerator ( StringBuilder & builder , MappingType & & mapping , char opening = ' @ ' , char closing = ' @ ' , char escape = ' \\ ' )
2020-10-23 18:36:56 +02:00
: m_builder ( builder )
2023-06-17 13:12:36 +01:00
, m_mapping ( move ( mapping ) )
2020-10-11 19:57:43 +02:00
, m_opening ( opening )
, m_closing ( closing )
2024-07-07 19:54:59 +01:00
, m_escape ( escape )
2020-10-11 19:57:43 +02:00
{
}
2022-02-22 19:20:08 +01:00
SourceGenerator ( SourceGenerator & & ) = default ;
2023-06-16 16:03:16 +02:00
// Move-assign is undefinable due to 'StringBuilder& m_builder;'
SourceGenerator & operator = ( SourceGenerator & & ) = delete ;
2022-02-22 19:20:08 +01:00
2023-08-21 16:42:48 +02:00
[ [ nodiscard ] ] SourceGenerator fork ( )
2023-06-17 13:12:36 +01:00
{
2023-08-21 16:42:48 +02:00
return SourceGenerator { m_builder , MUST ( m_mapping . clone ( ) ) , m_opening , m_closing } ;
2023-06-17 13:12:36 +01:00
}
2020-10-11 19:57:43 +02:00
2023-08-21 14:38:55 +02:00
void set ( StringView key , String value )
2022-08-23 16:40:39 +01: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-08-21 14:38:55 +02:00
m_mapping . set ( key , move ( value ) ) ;
2023-06-17 14:56:59 +01:00
}
String get ( StringView key ) const
2022-03-08 17:29:45 +00: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 19:57:43 +02:00
2020-10-23 18:36:56 +02:00
StringView as_string_view ( ) const { return m_builder . string_view ( ) ; }
2020-10-11 19:57:43 +02:00
2023-08-21 16:39:43 +02:00
void append ( StringView pattern )
2020-10-11 19:57:43 +02:00
{
GenericLexer lexer { pattern } ;
while ( ! lexer . is_eof ( ) ) {
2024-07-07 19:54:59 +01:00
m_builder . append ( lexer . consume_until ( [ & ] ( char ch ) { return ch = = m_opening | | ch = = m_escape ; } ) ) ;
if ( lexer . consume_specific ( m_escape ) ) {
if ( ! ( lexer . next_is ( m_opening ) | | lexer . next_is ( m_escape ) ) ) {
if ( lexer . is_eof ( ) )
warnln ( " Unexpected EOF while parsing escape sequence on SourceGenerator " ) ;
else
warnln ( " Invalid escape sequence found `{}{}` on SourceGenerator " , m_escape , lexer . peek ( ) ) ;
VERIFY_NOT_REACHED ( ) ;
}
m_builder . append ( lexer . consume ( ) ) ;
continue ;
}
2020-10-11 19:57:43 +02:00
if ( lexer . consume_specific ( m_opening ) ) {
2023-06-16 17:23:59 +01:00
auto const placeholder = lexer . consume_until ( m_closing ) ;
2020-10-11 19:57:43 +02:00
if ( ! lexer . consume_specific ( m_closing ) )
2021-02-23 20:42:32 +01:00
VERIFY_NOT_REACHED ( ) ;
2020-10-11 19:57:43 +02:00
2023-08-21 16:39:43 +02:00
m_builder . append ( get ( placeholder ) ) ;
2020-10-11 19:57:43 +02:00
} else {
2021-02-23 20:42:32 +01:00
VERIFY ( lexer . is_eof ( ) ) ;
2020-10-11 19:57:43 +02:00
}
}
}
2023-08-21 16:06:29 +02:00
void appendln ( StringView pattern )
2022-02-22 19:21:33 +01:00
{
2023-08-21 16:06:29 +02:00
append ( pattern ) ;
m_builder . append ( ' \n ' ) ;
2022-02-22 19:21:33 +01:00
}
2022-07-12 00:58:29 +00:00
template < size_t N >
2023-06-17 14:56:59 +01:00
String get ( char const ( & key ) [ N ] )
2022-07-12 00:58:29 +00:00
{
return get ( StringView { key , N - 1 } ) ;
}
template < size_t N >
2023-08-21 14:38:55 +02:00
void set ( char const ( & key ) [ N ] , String value )
2022-07-12 00:58:29 +00:00
{
2023-08-21 14:38:55 +02:00
set ( StringView { key , N - 1 } , value ) ;
2022-07-12 00:58:29 +00:00
}
2023-06-17 14:56:59 +01:00
template < size_t N >
2023-08-21 16:39:43 +02:00
void append ( char const ( & pattern ) [ N ] )
2023-06-17 14:56:59 +01:00
{
2023-08-21 16:39:43 +02:00
append ( StringView { pattern , N - 1 } ) ;
2023-06-17 14:56:59 +01:00
}
2022-07-12 00:58:29 +00:00
template < size_t N >
2023-08-21 16:06:29 +02:00
void appendln ( char const ( & pattern ) [ N ] )
2022-07-12 00:58:29 +00:00
{
2023-08-21 16:06:29 +02:00
appendln ( StringView { pattern , N - 1 } ) ;
2022-07-12 00:58:29 +00:00
}
2023-06-17 14:56:03 +01:00
// FIXME: These are deprecated.
2023-12-16 17:49:34 +03:30
void set ( StringView key , ByteString value )
2023-06-17 14:56:03 +01:00
{
2023-12-16 17:49:34 +03:30
set ( key , MUST ( String : : from_byte_string ( value ) ) ) ;
2023-06-17 14:56:03 +01:00
}
2022-07-12 00:58:29 +00:00
template < size_t N >
2023-12-16 17:49:34 +03:30
void set ( char const ( & key ) [ N ] , ByteString value )
2022-07-12 00:58:29 +00:00
{
2023-06-17 14:56:03 +01:00
set ( StringView { key , N - 1 } , value ) ;
2022-07-12 00:58:29 +00:00
}
2020-10-11 19:57:43 +02:00
private :
2020-10-23 18:36:56 +02:00
StringBuilder & m_builder ;
2020-10-11 19:57:43 +02:00
MappingType m_mapping ;
2024-07-07 19:54:59 +01:00
char m_opening { ' @ ' } ;
char m_closing { ' @ ' } ;
char m_escape { ' \\ ' } ;
2020-10-11 19:57:43 +02:00
} ;
}
2022-11-26 12:18:30 +01:00
# if USING_AK_GLOBALLY
2020-10-11 19:57:43 +02:00
using AK : : SourceGenerator ;
2022-11-26 12:18:30 +01:00
# endif