2011-02-09 17:31:16 +13:00
< ? php
/*!* !insert_autogen_warning */
/*!* ! silent
This is the uncompiled parser for the SilverStripe template language , PHP with special comments that define the parser .
It gets run through the php - peg parser compiler to have those comments turned into code that match parts of the template language ,
producing the executable version SSTemplateParser . php
2011-02-21 13:45:56 +13:00
To recompile after changing this file , run this from the 'sapphire/core' directory via command line :
php ../ thirdparty / php - peg / cli . php SSTemplateParser . php . inc > SSTemplateParser . php
2011-02-09 17:31:16 +13:00
See the php - peg docs for more information on the parser format , and how to convert this file into SSTemplateParser . php
TODO :
Template comments - <%-- --%>
$Iteration
Partial cache blocks
i18n - we dont support then deprecated _t () or sprintf ( _t ()) methods ; or the new <% t %> block yet
Add with and loop blocks
Add Up and Top
More error detection ?
This comment will not appear in the output
*/
// We want this to work when run by hand too
if ( defined ( THIRDPARTY_PATH )) {
2011-12-04 12:54:07 +01:00
require_once ( THIRDPARTY_PATH . '/php-peg/Parser.php' );
2011-02-09 17:31:16 +13:00
}
else {
$base = dirname ( __FILE__ );
2011-12-04 12:54:07 +01:00
require_once ( $base . '/../thirdparty/php-peg/Parser.php' );
2011-02-09 17:31:16 +13:00
}
/**
This is the exception raised when failing to parse a template . Note that we don 't currently do any static analysis, so we can' t know
if the template will run , just if it 's malformed. It also won' t catch mistakes that still look valid .
*/
class SSTemplateParseException extends Exception {
function __construct ( $message , $parser ) {
$prior = substr ( $parser -> string , 0 , $parser -> pos );
preg_match_all ( '/\r\n|\r|\n/' , $prior , $matches );
$line = count ( $matches [ 0 ]) + 1 ;
parent :: __construct ( " Parse error in template on line $line . Error was: $message " );
}
}
/**
This is the parser for the SilverStripe template language . It gets called on a string and uses a php - peg parser to match
that string against the language structure , building up the PHP code to execute that structure as it parses
2011-02-15 10:03:42 +13:00
The $result array that is built up as part of the parsing ( see thirdparty / php - peg / README . md for more on how parsers
build results ) has one special member , 'php' , which contains the php equivalent of that part of the template tree .
Some match rules generate alternate php , or other variations , so check the per - match documentation too .
Terms used :
Marked : A string or lookup in the template that has been explictly marked as such - lookups by prepending with " $ "
( like $Foo . Bar ), strings by wrapping with single or double quotes ( 'Foo' or " Foo " )
Bare : The opposite of marked . An argument that has to has it ' s type inferred by usage and 2.4 defaults .
Example of using a bare argument for a loop block : <% loop Foo %>
Block : One of two SS template structures . The special characters " <% " and " %> " are used to wrap the opening and
( required or forbidden depending on which block exactly ) closing block marks .
Open Block : An SS template block that doesn ' t wrap any content or have a closing end tag ( in fact , a closing end tag is
forbidden )
Closed Block : An SS template block that wraps content , and requires a counterpart <% end_blockname %> tag
2011-02-09 17:31:16 +13:00
*/
class SSTemplateParser extends Parser {
2011-02-15 10:03:42 +13:00
/**
* @ var bool - Set true by SSTemplateParser :: compileString if the template should include comments intended
* for debugging ( template source , included files , etc )
*/
2011-02-09 17:31:16 +13:00
protected $includeDebuggingComments = false ;
2011-02-28 16:12:41 +13:00
/**
* Override the function that constructs the result arrays to also prepare a 'php' item in the array
*/
function construct ( $matchrule , $name , $arguments = null ) {
$res = parent :: construct ( $matchrule , $name , $arguments );
if ( ! isset ( $res [ 'php' ])) $res [ 'php' ] = '' ;
return $res ;
}
2011-02-09 17:31:16 +13:00
/*!* SSTemplateParser
2011-02-28 16:12:41 +13:00
# Template is any structurally-complete portion of template (a full nested level in other words). It's the primary matcher,
# and is used by all enclosing blocks, as well as a base for the top level
Template : ( Comment | If | Require | CacheBlock | UncachedBlock | OldI18NTag | ClosedBlock | OpenBlock | MalformedBlock | Injection | Text ) +
*/
function Template_STR ( & $res , $sub ) {
$res [ 'php' ] .= $sub [ 'php' ] . PHP_EOL ;
}
/*!*
2011-02-09 17:31:16 +13:00
Word : / [ A - Za - z_ ] [ A - Za - z0 - 9_ ] * /
Number : / [ 0 - 9 ] + /
Value : / [ A - Za - z0 - 9_ ] + /
2011-02-15 10:03:42 +13:00
# CallArguments is a list of one or more comma seperated "arguments" (lookups or strings, either bare or marked)
# as passed to a Call within brackets
CallArguments : : Argument ( < " , " < : Argument ) *
*/
2011-02-09 17:31:16 +13:00
2011-02-15 10:03:42 +13:00
/**
* Values are bare words in templates , but strings in PHP . We rely on PHP ' s type conversion to back - convert strings
* to numbers when needed .
*/
function CallArguments_Argument ( & $res , $sub ) {
2011-02-28 16:12:41 +13:00
if ( ! empty ( $res [ 'php' ])) $res [ 'php' ] .= ', ' ;
2011-02-09 17:31:16 +13:00
2011-02-18 17:06:11 +13:00
$res [ 'php' ] .= ( $sub [ 'ArgumentMode' ] == 'default' ) ? $sub [ 'string_php' ] : str_replace ( '$$FINAL' , 'XML_val' , $sub [ 'php' ]);
2011-02-09 17:31:16 +13:00
}
/*!*
2011-02-15 10:03:42 +13:00
# Call is a php-style function call, e.g. Method(Argument, ...). Unlike PHP, the brackets are optional if no
# arguments are passed
Call : Method : Word ( " ( " < : CallArguments ? > " ) " ) ?
# A lookup is a lookup of a value on the current scope object. It's a sequence of calls seperated by "." characters
# This final call in the sequence needs handling specially, as different structures need different sorts of values,
# which require a different final method to be called to get the right return value
2011-02-09 17:31:16 +13:00
LookupStep : : Call & " . "
LastLookupStep : : Call
Lookup : LookupStep ( " . " LookupStep ) * " . " LastLookupStep | LastLookupStep
*/
function Lookup__construct ( & $res ) {
2011-02-18 17:06:11 +13:00
$res [ 'php' ] = '$scope' ;
2011-02-09 17:31:16 +13:00
$res [ 'LookupSteps' ] = array ();
}
2011-02-15 10:03:42 +13:00
/**
* The basic generated PHP of LookupStep and LastLookupStep is the same , except that LookupStep calls 'obj' to
* get the next ViewableData in the sequence , and LastLookupStep calls different methods ( XML_val , hasValue , obj )
* depending on the context the lookup is used in .
*/
2011-02-09 17:31:16 +13:00
function Lookup_AddLookupStep ( & $res , $sub , $method ) {
$res [ 'LookupSteps' ][] = $sub ;
$property = $sub [ 'Call' ][ 'Method' ][ 'text' ];
2011-02-15 10:03:42 +13:00
if ( isset ( $sub [ 'Call' ][ 'CallArguments' ]) && $arguments = $sub [ 'Call' ][ 'CallArguments' ][ 'php' ]) {
2011-02-09 17:31:16 +13:00
$res [ 'php' ] .= " -> $method (' $property ', array( $arguments ), true) " ;
}
else {
$res [ 'php' ] .= " -> $method (' $property ', null, true) " ;
}
}
function Lookup_LookupStep ( & $res , $sub ) {
$this -> Lookup_AddLookupStep ( $res , $sub , 'obj' );
}
function Lookup_LastLookupStep ( & $res , $sub ) {
2011-02-18 17:06:11 +13:00
$this -> Lookup_AddLookupStep ( $res , $sub , '$$FINAL' );
2011-02-09 17:31:16 +13:00
}
/*!*
2011-02-15 10:03:42 +13:00
# Injections are where, outside of a block, a value needs to be inserted into the output. You can either
# just do $Foo, or {$Foo} if the surrounding text would cause a problem (e.g. {$Foo}Bar)
2011-02-09 17:31:16 +13:00
SimpleInjection : '$' : Lookup
BracketInjection : '{$' : Lookup " } "
Injection : BracketInjection | SimpleInjection
*/
function Injection_STR ( & $res , $sub ) {
2011-02-18 17:06:11 +13:00
$res [ 'php' ] = '$val .= ' . str_replace ( '$$FINAL' , 'XML_val' , $sub [ 'Lookup' ][ 'php' ]) . ';' ;
2011-02-09 17:31:16 +13:00
}
/*!*
2011-02-15 10:03:42 +13:00
# Inside a block's arguments you can still use the same format as a simple injection ($Foo). In this case
# it marks the argument as being a lookup, not a string (if it was bare it might still be used as a lookup,
# but that depends on where it's used, a la 2.4)
DollarMarkedLookup : SimpleInjection
2011-02-09 17:31:16 +13:00
*/
function DollarMarkedLookup_STR ( & $res , $sub ) {
$res [ 'Lookup' ] = $sub [ 'Lookup' ];
}
/*!*
2011-02-15 10:03:42 +13:00
# Inside a block's arguments you can explictly mark a string by surrounding it with quotes (single or double,
# but they must be matching). If you do, inside the quote you can escape any character, but the only character
# that _needs_ escaping is the matching closing quote
2011-02-09 17:31:16 +13:00
QuotedString : q :/ [ '"]/ String:/ (\\\\ | \\. | [^$q\\])* / ' $q '
2011-02-15 10:03:42 +13:00
# In order to support 2.4's base syntax, we also need to detect free strings - strings not surrounded by
# quotes, and containing spaces or punctuation, but supported as a single string. We support almost as flexible
# a string as 2.4 - we don't attempt to determine the closing character by context, but just break on any character
# which, in some context, would indicate the end of a free string, regardless of if we're actually in that context
# or not
2011-02-09 17:31:16 +13:00
FreeString : / [ ^ ,) %!=|& ] +/
2011-02-15 10:03:42 +13:00
# An argument - either a marked value, or a bare value, prefering lookup matching on the bare value over freestring
# matching as long as that would give a successful parse
2011-02-09 17:31:16 +13:00
Argument :
: DollarMarkedLookup |
: QuotedString |
: Lookup ! ( < FreeString ) |
: FreeString
*/
2011-02-15 10:03:42 +13:00
/**
* If we get a bare value , we don ' t know enough to determine exactly what php would be the translation , because
* we don ' t know if the position of use indicates a lookup or a string argument .
*
* Instead , we record 'ArgumentMode' as a member of this matches results node , which can be :
* - lookup if this argument was unambiguously a lookup ( marked as such )
* - string is this argument was unambiguously a string ( marked as such , or impossible to parse as lookup )
* - default if this argument needs to be handled as per 2.4
*
* In the case of 'default' , there is no php member of the results node , but instead 'lookup_php' , which
* should be used by the parent if the context indicates a lookup , and 'string_php' which should be used
* if the context indicates a string
*/
2011-02-09 17:31:16 +13:00
function Argument_DollarMarkedLookup ( & $res , $sub ) {
$res [ 'ArgumentMode' ] = 'lookup' ;
$res [ 'php' ] = $sub [ 'Lookup' ][ 'php' ];
}
function Argument_QuotedString ( & $res , $sub ) {
$res [ 'ArgumentMode' ] = 'string' ;
2011-02-21 17:44:46 +13:00
$res [ 'php' ] = " ' " . str_replace ( " ' " , " \\ ' " , $sub [ 'String' ][ 'text' ]) . " ' " ;
2011-02-09 17:31:16 +13:00
}
function Argument_Lookup ( & $res , $sub ) {
if ( count ( $sub [ 'LookupSteps' ]) == 1 && ! isset ( $sub [ 'LookupSteps' ][ 0 ][ 'Call' ][ 'Arguments' ])) {
$res [ 'ArgumentMode' ] = 'default' ;
$res [ 'lookup_php' ] = $sub [ 'php' ];
$res [ 'string_php' ] = " ' " . $sub [ 'LookupSteps' ][ 0 ][ 'Call' ][ 'Method' ][ 'text' ] . " ' " ;
}
else {
$res [ 'ArgumentMode' ] = 'lookup' ;
$res [ 'php' ] = $sub [ 'php' ];
}
}
function Argument_FreeString ( & $res , $sub ) {
$res [ 'ArgumentMode' ] = 'string' ;
2011-02-21 17:44:46 +13:00
$res [ 'php' ] = " ' " . str_replace ( " ' " , " \\ ' " , $sub [ 'text' ]) . " ' " ;
2011-02-09 17:31:16 +13:00
}
/*!*
2011-02-15 10:03:42 +13:00
# if and else_if blocks allow basic comparisons between arguments
2011-02-09 17:31:16 +13:00
ComparisonOperator : " == " | " != " | " = "
Comparison : Argument < ComparisonOperator > Argument
*/
function Comparison_Argument ( & $res , $sub ) {
if ( $sub [ 'ArgumentMode' ] == 'default' ) {
2011-02-28 16:12:41 +13:00
if ( ! empty ( $res [ 'php' ])) $res [ 'php' ] .= $sub [ 'string_php' ];
2011-02-18 17:06:11 +13:00
else $res [ 'php' ] = str_replace ( '$$FINAL' , 'XML_val' , $sub [ 'lookup_php' ]);
2011-02-09 17:31:16 +13:00
}
else {
2011-02-18 17:06:11 +13:00
$res [ 'php' ] .= str_replace ( '$$FINAL' , 'XML_val' , $sub [ 'php' ]);
2011-02-09 17:31:16 +13:00
}
}
function Comparison_ComparisonOperator ( & $res , $sub ) {
$res [ 'php' ] .= ( $sub [ 'text' ] == '=' ? '==' : $sub [ 'text' ]);
}
/*!*
2011-02-15 10:03:42 +13:00
# If a comparison operator is not used in an if or else_if block, then the statement is a 'presence check',
# which checks if the argument given is present or not. For explicit strings (which were not allowed in 2.4)
# this falls back to simple truthiness check
2011-02-21 15:24:14 +13:00
PresenceCheck : ( Not : 'not' < ) ? Argument
2011-02-09 17:31:16 +13:00
*/
2011-02-21 15:24:14 +13:00
function PresenceCheck_Not ( & $res , $sub ) {
$res [ 'php' ] = '!' ;
}
2011-02-09 17:31:16 +13:00
function PresenceCheck_Argument ( & $res , $sub ) {
if ( $sub [ 'ArgumentMode' ] == 'string' ) {
2011-02-21 15:24:14 +13:00
$res [ 'php' ] .= '((bool)' . $sub [ 'php' ] . ')' ;
2011-02-09 17:31:16 +13:00
}
else {
$php = ( $sub [ 'ArgumentMode' ] == 'default' ? $sub [ 'lookup_php' ] : $sub [ 'php' ]);
// TODO: kinda hacky - maybe we need a way to pass state down the parse chain so
// Lookup_LastLookupStep and Argument_BareWord can produce hasValue instead of XML_val
2011-02-21 15:24:14 +13:00
$res [ 'php' ] .= str_replace ( '$$FINAL' , 'hasValue' , $php );
2011-02-09 17:31:16 +13:00
}
}
/*!*
2011-02-15 10:03:42 +13:00
# if and else_if arguments are a series of presence checks and comparisons, optionally seperated by boolean
# operators
2011-02-09 17:31:16 +13:00
IfArgumentPortion : Comparison | PresenceCheck
*/
function IfArgumentPortion_STR ( & $res , $sub ) {
$res [ 'php' ] = $sub [ 'php' ];
}
2011-02-15 10:03:42 +13:00
/*!*
# if and else_if arguments can be combined via these two boolean operators. No precendence overriding is supported
2011-02-09 17:31:16 +13:00
BooleanOperator : " || " | " && "
2011-02-15 10:03:42 +13:00
# This is the combination of the previous if and else_if argument portions
2011-02-09 17:31:16 +13:00
IfArgument : : IfArgumentPortion ( < : BooleanOperator < : IfArgumentPortion ) *
*/
function IfArgument_IfArgumentPortion ( & $res , $sub ) {
$res [ 'php' ] .= $sub [ 'php' ];
}
function IfArgument_BooleanOperator ( & $res , $sub ) {
$res [ 'php' ] .= $sub [ 'text' ];
}
/*!*
2011-02-15 10:03:42 +13:00
# ifs are handled seperately from other closed block tags, because (A) their structure is different - they
# can have else_if and else tags in between the if tag and the end_if tag, and (B) they have a different
# argument structure to every other block
2011-02-28 16:12:41 +13:00
IfPart : '<%' < 'if' [ : IfArgument > '%>' Template : $TemplateMatcher ?
ElseIfPart : '<%' < 'else_if' [ : IfArgument > '%>' Template : $TemplateMatcher
ElsePart : '<%' < 'else' > '%>' Template : $TemplateMatcher
2011-02-09 17:31:16 +13:00
If : IfPart ElseIfPart * ElsePart ? '<%' < 'end_if' > '%>'
*/
function If_IfPart ( & $res , $sub ) {
$res [ 'php' ] =
'if (' . $sub [ 'IfArgument' ][ 'php' ] . ') { ' . PHP_EOL .
2011-02-21 16:19:10 +13:00
( isset ( $sub [ 'Template' ]) ? $sub [ 'Template' ][ 'php' ] : '' ) . PHP_EOL .
2011-02-09 17:31:16 +13:00
'}' ;
}
function If_ElseIfPart ( & $res , $sub ) {
$res [ 'php' ] .=
'else if (' . $sub [ 'IfArgument' ][ 'php' ] . ') { ' . PHP_EOL .
2011-02-21 16:19:10 +13:00
$sub [ 'Template' ][ 'php' ] . PHP_EOL .
2011-02-09 17:31:16 +13:00
'}' ;
}
function If_ElsePart ( & $res , $sub ) {
$res [ 'php' ] .=
'else { ' . PHP_EOL .
$sub [ 'Template' ][ 'php' ] . PHP_EOL .
'}' ;
}
/*!*
2011-02-15 10:03:42 +13:00
# The require block is handled seperately to the other open blocks as the argument syntax is different
# - must have one call style argument, must pass arguments to that call style argument
Require : '<%' < 'require' [ Call : ( Method : Word " ( " < : CallArguments > " ) " ) > '%>'
2011-02-09 17:31:16 +13:00
*/
function Require_Call ( & $res , $sub ) {
2011-02-15 10:03:42 +13:00
$res [ 'php' ] = " Requirements:: " . $sub [ 'Method' ][ 'text' ] . '(' . $sub [ 'CallArguments' ][ 'php' ] . ');' ;
2011-02-09 17:31:16 +13:00
}
2011-02-21 17:44:46 +13:00
2011-02-28 16:12:41 +13:00
/*!*
# Cache block arguments don't support free strings
CacheBlockArgument :
! ( " if " | " unless " )
(
: DollarMarkedLookup |
: QuotedString |
: Lookup
)
*/
function CacheBlockArgument_DollarMarkedLookup ( & $res , $sub ) {
$res [ 'php' ] = $sub [ 'Lookup' ][ 'php' ];
}
function CacheBlockArgument_QuotedString ( & $res , $sub ) {
$res [ 'php' ] = " ' " . str_replace ( " ' " , " \\ ' " , $sub [ 'String' ][ 'text' ]) . " ' " ;
}
function CacheBlockArgument_Lookup ( & $res , $sub ) {
$res [ 'php' ] = $sub [ 'php' ];
}
/*!*
# Collects the arguments passed in to be part of the key of a cacheblock
CacheBlockArguments : CacheBlockArgument ( < " , " < CacheBlockArgument ) *
*/
function CacheBlockArguments_CacheBlockArgument ( & $res , $sub ) {
if ( ! empty ( $res [ 'php' ])) $res [ 'php' ] .= " .'_'. " ;
else $res [ 'php' ] = '' ;
$res [ 'php' ] .= str_replace ( '$$FINAL' , 'XML_val' , $sub [ 'php' ]);
}
/*!*
# CacheBlockTemplate is the same as Template, but doesn't include cache blocks (because they're handled seperately)
CacheBlockTemplate extends Template ( TemplateMatcher = CacheRestrictedTemplate ); CacheBlock | UncachedBlock | => ''
*/
/*!*
UncachedBlock :
'<%' < " uncached " < CacheBlockArguments ? ( < Conditional : ( " if " | " unless " ) > Condition : IfArgument ) ? > '%>'
Template : $TemplateMatcher ?
'<%' < 'end_' ( " uncached " | " cached " | " cacheblock " ) > '%>'
*/
function UncachedBlock_Template ( & $res , $sub ){
$res [ 'php' ] = $sub [ 'php' ];
}
/*!*
# CacheRestrictedTemplate is the same as Template, but doesn't allow cache blocks
CacheRestrictedTemplate extends Template
*/
function CacheRestrictedTemplate_CacheBlock ( & $res , $sub ) {
throw new SSTemplateParseException ( 'You cant have cache blocks nested within with, loop or control blocks that are within cache blocks' , $this );
}
function CacheRestrictedTemplate_UncachedBlock ( & $res , $sub ) {
throw new SSTemplateParseException ( 'You cant have uncache blocks nested within with, loop or control blocks that are within cache blocks' , $this );
}
/*!*
# The partial caching block
CacheBlock :
'<%' < CacheTag : ( " cached " | " cacheblock " ) < ( CacheBlockArguments ) ? ( < Conditional : ( " if " | " unless " ) > Condition : IfArgument ) ? > '%>'
( CacheBlock | UncachedBlock | CacheBlockTemplate ) *
'<%' < 'end_' ( " cached " | " uncached " | " cacheblock " ) > '%>'
*/
function CacheBlock__construct ( & $res ){
$res [ 'subblocks' ] = 0 ;
}
function CacheBlock_CacheBlockArguments ( & $res , $sub ){
$res [ 'key' ] = ! empty ( $sub [ 'php' ]) ? $sub [ 'php' ] : '' ;
}
function CacheBlock_Condition ( & $res , $sub ){
$res [ 'condition' ] = ( $res [ 'Conditional' ][ 'text' ] == 'if' ? '(' : '!(' ) . $sub [ 'php' ] . ') && ' ;
}
function CacheBlock_CacheBlock ( & $res , $sub ){
$res [ 'php' ] .= $sub [ 'php' ];
}
function CacheBlock_UncachedBlock ( & $res , $sub ){
$res [ 'php' ] .= $sub [ 'php' ];
}
function CacheBlock_CacheBlockTemplate ( & $res , $sub ){
// Get the block counter
$block = ++ $res [ 'subblocks' ];
// Build the key for this block from the passed cache key, the block index, and the sha hash of the template itself
$key = " ' " . sha1 ( $sub [ 'php' ]) . ( isset ( $res [ 'key' ]) && $res [ 'key' ] ? " _'.sha1( " . $res [ 'key' ] . " ) " : " ' " ) . " .'_ $block ' " ;
2011-12-23 10:38:37 +13:00
$key = preg_replace ( '/[^a-zA-Z0-9_]/' , '_' , $key );
2011-02-28 16:12:41 +13:00
// Get any condition
$condition = isset ( $res [ 'condition' ]) ? $res [ 'condition' ] : '' ;
$res [ 'php' ] .= 'if (' . $condition . '($partial = $cache->load(' . $key . '))) $val .= $partial;' . PHP_EOL ;
$res [ 'php' ] .= 'else { $oldval = $val; $val = "";' . PHP_EOL ;
$res [ 'php' ] .= $sub [ 'php' ] . PHP_EOL ;
$res [ 'php' ] .= $condition . ' $cache->save($val); $val = $oldval . $val;' . PHP_EOL ;
$res [ 'php' ] .= '}' ;
}
2011-02-21 17:44:46 +13:00
/*!*
# Deprecated old-style i18n _t and sprintf(_t block tags. We support a slightly more flexible version than we used
# to, but just because it's easier to do so. It's strongly recommended to use the new syntax
# This is the core used by both syntaxes, without the block start & end tags
OldTPart : " _t " < " ( " < QuotedString ( < " , " < CallArguments ) ? > " ) "
*/
function OldTPart__construct ( & $res ) {
$res [ 'php' ] = " _t( " ;
}
function OldTPart_QuotedString ( & $res , $sub ) {
$entity = $sub [ 'String' ][ 'text' ];
if ( strpos ( $entity , '.' ) === false ) {
$res [ 'php' ] .= " \$ scope->XML_val('I18NNamespace').'. $entity ' " ;
}
else {
$res [ 'php' ] .= " ' $entity ' " ;
}
}
function OldTPart_CallArguments ( & $res , $sub ) {
$res [ 'php' ] .= ',' . $sub [ 'php' ];
}
function OldTPart__finalise ( & $res ) {
$res [ 'php' ] .= ')' ;
}
/*!*
# This is the old <% _t() %> tag
OldTTag : " <% " < OldTPart > " %> "
*/
function OldTTag_OldTPart ( & $res , $sub ) {
$res [ 'php' ] = $sub [ 'php' ];
}
/*!*
# This is the old <% sprintf(_t()) %> tag
OldSprintfTag : " <% " < " sprintf " < " ( " < OldTPart < " , " < CallArguments > " ) " > " %> "
*/
function OldSprintfTag__construct ( & $res ) {
$res [ 'php' ] = " sprintf( " ;
}
function OldSprintfTag_OldTPart ( & $res , $sub ) {
$res [ 'php' ] .= $sub [ 'php' ];
}
function OldSprintfTag_CallArguments ( & $res , $sub ) {
$res [ 'php' ] .= ',' . $sub [ 'php' ] . ')' ;
}
/*!*
# This matches either the old style sprintf(_t()) or _t() tags. As well as including the output portion of the
# php, this rule combines all the old i18n stuff into a single match rule to make it easy to not support these tags later
OldI18NTag : OldSprintfTag | OldTTag
*/
function OldI18NTag_STR ( & $res , $sub ) {
$res [ 'php' ] = '$val .= ' . $sub [ 'php' ] . ';' ;
}
2011-02-09 17:31:16 +13:00
/*!*
2011-02-15 10:03:42 +13:00
# To make the block support reasonably extendable, we don't explicitly define each closed block and it's structure,
# but instead match against a generic <% block_name argument, ... %> pattern. Each argument is left as per the
# output of the Argument matcher, and the handler (see the PHPDoc block later for more on this) is responsible
# for pulling out the info required
2011-02-09 17:31:16 +13:00
BlockArguments : : Argument ( < " , " < : Argument ) *
2011-02-15 10:03:42 +13:00
# NotBlockTag matches against any word that might come after a "<%" that the generic open and closed block handlers
# shouldn't attempt to match against, because they're handled by more explicit matchers
2011-02-28 16:12:41 +13:00
NotBlockTag : " end_ " | (( " if " | " else_if " | " else " | " require " | " cached " | " uncached " | " cacheblock " ) ] )
2011-02-09 17:31:16 +13:00
2011-02-15 10:03:42 +13:00
# Match against closed blocks - blocks with an opening and a closing tag that surround some internal portion of
# template
2011-02-28 16:12:41 +13:00
ClosedBlock : '<%' < ! NotBlockTag BlockName : Word ( [ : BlockArguments ] ) ? > Zap : '%>' Template : $TemplateMatcher ? '<%' < 'end_' '$BlockName' > '%>'
2011-02-09 17:31:16 +13:00
*/
2011-02-15 10:03:42 +13:00
/**
* As mentioned in the parser comment , block handling is kept fairly generic for extensibility . The match rule
* builds up two important elements in the match result array :
* 'ArgumentCount' - how many arguments were passed in the opening tag
* 'Arguments' an array of the Argument match rule result arrays
*
* Once a block has successfully been matched against , it will then look for the actual handler , which should
2011-04-15 19:35:30 +10:00
* be on this class ( either defined or extended on ) as ClosedBlock_Handler_Name ( & $res ), where Name is the
2011-02-15 10:03:42 +13:00
* tag name , first letter captialized ( i . e Control , Loop , With , etc ) .
*
* This function will be called with the match rule result array as it ' s first argument . It should return
* the php result of this block as it ' s return value , or throw an error if incorrect arguments were passed .
*/
2011-02-09 17:31:16 +13:00
function ClosedBlock__construct ( & $res ) {
$res [ 'ArgumentCount' ] = 0 ;
}
function ClosedBlock_BlockArguments ( & $res , $sub ) {
if ( isset ( $sub [ 'Argument' ][ 'ArgumentMode' ])) {
$res [ 'Arguments' ] = array ( $sub [ 'Argument' ]);
$res [ 'ArgumentCount' ] = 1 ;
}
else {
$res [ 'Arguments' ] = $sub [ 'Argument' ];
$res [ 'ArgumentCount' ] = count ( $res [ 'Arguments' ]);
}
}
function ClosedBlock__finalise ( & $res ) {
$blockname = $res [ 'BlockName' ][ 'text' ];
2011-12-17 17:35:26 +13:00
$method = 'ClosedBlock_Handle_' . $blockname ;
2011-02-09 17:31:16 +13:00
if ( method_exists ( $this , $method )) $res [ 'php' ] = $this -> $method ( $res );
else {
throw new SSTemplateParseException ( 'Unknown closed block "' . $blockname . '" encountered. Perhaps you are not supposed to close this block, or have mis-spelled it?' , $this );
}
}
2011-02-15 10:03:42 +13:00
/**
2011-02-18 17:06:11 +13:00
* This is an example of a block handler function . This one handles the loop tag .
2011-02-15 10:03:42 +13:00
*/
2011-02-18 17:06:11 +13:00
function ClosedBlock_Handle_Loop ( & $res ) {
2012-02-11 15:30:46 +13:00
if ( $res [ 'ArgumentCount' ] > 1 ) {
2011-02-09 17:31:16 +13:00
throw new SSTemplateParseException ( 'Either no or too many arguments in control block. Must be one argument only.' , $this );
}
2012-02-11 15:30:46 +13:00
//loop without arguments loops on the current scope
if ( $res [ 'ArgumentCount' ] == 0 ) {
$on = '$scope->obj(\'Up\', null, true)->obj(\'Foo\', null, true)' ;
} else { //loop in the normal way
$arg = $res [ 'Arguments' ][ 0 ];
if ( $arg [ 'ArgumentMode' ] == 'string' ) {
throw new SSTemplateParseException ( 'Control block cant take string as argument.' , $this );
}
$on = str_replace ( '$$FINAL' , 'obj' , ( $arg [ 'ArgumentMode' ] == 'default' ) ? $arg [ 'lookup_php' ] : $arg [ 'php' ]);
2011-02-09 17:31:16 +13:00
}
2012-02-11 15:30:46 +13:00
return
2011-02-18 17:06:11 +13:00
$on . '; $scope->pushScope(); while (($key = $scope->next()) !== false) {' . PHP_EOL .
$res [ 'Template' ][ 'php' ] . PHP_EOL .
'}; $scope->popScope(); ' ;
}
/**
* The deprecated closed block handler for control blocks
* @ deprecated
*/
function ClosedBlock_Handle_Control ( & $res ) {
2011-10-29 12:02:11 +13:00
Deprecation :: notice ( '3.1' , 'Use <% with %> or <% loop %> instead.' );
2011-02-18 17:06:11 +13:00
return $this -> ClosedBlock_Handle_Loop ( $res );
}
/**
* The closed block handler for with blocks
*/
function ClosedBlock_Handle_With ( & $res ) {
if ( $res [ 'ArgumentCount' ] != 1 ) {
throw new SSTemplateParseException ( 'Either no or too many arguments in with block. Must be one argument only.' , $this );
}
$arg = $res [ 'Arguments' ][ 0 ];
if ( $arg [ 'ArgumentMode' ] == 'string' ) {
throw new SSTemplateParseException ( 'Control block cant take string as argument.' , $this );
}
$on = str_replace ( '$$FINAL' , 'obj' , ( $arg [ 'ArgumentMode' ] == 'default' ) ? $arg [ 'lookup_php' ] : $arg [ 'php' ]);
2011-02-09 17:31:16 +13:00
return
2011-02-18 17:06:11 +13:00
$on . '; $scope->pushScope();' . PHP_EOL .
2011-02-09 17:31:16 +13:00
$res [ 'Template' ][ 'php' ] . PHP_EOL .
2011-02-18 17:06:11 +13:00
'; $scope->popScope(); ' ;
2011-02-09 17:31:16 +13:00
}
/*!*
2011-02-15 10:03:42 +13:00
# Open blocks are handled in the same generic manner as closed blocks. There is no need to define which blocks
# are which - closed is tried first, and if no matching end tag is found, open is tried next
2011-02-14 16:52:01 +13:00
OpenBlock : '<%' < ! NotBlockTag BlockName : Word ( [ : BlockArguments ] ) ? > '%>'
2011-02-09 17:31:16 +13:00
*/
function OpenBlock__construct ( & $res ) {
$res [ 'ArgumentCount' ] = 0 ;
}
function OpenBlock_BlockArguments ( & $res , $sub ) {
if ( isset ( $sub [ 'Argument' ][ 'ArgumentMode' ])) {
$res [ 'Arguments' ] = array ( $sub [ 'Argument' ]);
$res [ 'ArgumentCount' ] = 1 ;
}
else {
$res [ 'Arguments' ] = $sub [ 'Argument' ];
$res [ 'ArgumentCount' ] = count ( $res [ 'Arguments' ]);
}
}
function OpenBlock__finalise ( & $res ) {
2011-02-14 16:52:01 +13:00
$blockname = $res [ 'BlockName' ][ 'text' ];
2011-02-09 17:31:16 +13:00
2011-12-17 17:35:26 +13:00
$method = 'OpenBlock_Handle_' . $blockname ;
2011-02-09 17:31:16 +13:00
if ( method_exists ( $this , $method )) $res [ 'php' ] = $this -> $method ( $res );
else {
throw new SSTemplateParseException ( 'Unknown open block "' . $blockname . '" encountered. Perhaps you missed the closing tag or have mis-spelled it?' , $this );
}
}
2011-02-15 10:03:42 +13:00
/**
* This is an open block handler , for the <% include %> tag
*/
2011-02-09 17:31:16 +13:00
function OpenBlock_Handle_Include ( & $res ) {
if ( $res [ 'ArgumentCount' ] != 1 ) throw new SSTemplateParseException ( 'Include takes exactly one argument' , $this );
$arg = $res [ 'Arguments' ][ 0 ];
$php = ( $arg [ 'ArgumentMode' ] == 'default' ) ? $arg [ 'string_php' ] : $arg [ 'php' ];
if ( $this -> includeDebuggingComments ) { // Add include filename comments on dev sites
return
2011-12-17 17:35:26 +13:00
'$val .= \'<!-- include ' . addslashes ( $php ) . ' -->\';' . " \n " .
'$val .= SSViewer::execute_template(' . $php . ', $scope->getItem());' . " \n " .
'$val .= \'<!-- end include ' . addslashes ( $php ) . ' -->\';' . " \n " ;
2011-02-09 17:31:16 +13:00
}
else {
return
2011-02-18 17:06:11 +13:00
'$val .= SSViewer::execute_template(' . $php . ', $scope->getItem());' . " \n " ;
2011-02-09 17:31:16 +13:00
}
}
2011-02-15 10:03:42 +13:00
/**
* This is an open block handler , for the <% debug %> utility tag
*/
2011-02-09 17:31:16 +13:00
function OpenBlock_Handle_Debug ( & $res ) {
2011-02-18 17:06:11 +13:00
if ( $res [ 'ArgumentCount' ] == 0 ) return '$scope->debug();' ;
2011-02-09 17:31:16 +13:00
else if ( $res [ 'ArgumentCount' ] == 1 ) {
$arg = $res [ 'Arguments' ][ 0 ];
if ( $arg [ 'ArgumentMode' ] == 'string' ) return 'Debug::show(' . $arg [ 'php' ] . ');' ;
$php = ( $arg [ 'ArgumentMode' ] == 'default' ) ? $arg [ 'lookup_php' ] : $arg [ 'php' ];
return '$val .= Debug::show(' . str_replace ( 'FINALGET!' , 'cachedCall' , $php ) . ');' ;
}
else {
throw new SSTemplateParseException ( 'Debug takes 0 or 1 argument only.' , $this );
}
}
2011-02-15 10:03:42 +13:00
/**
* This is an open block handler , for the <% base_tag %> tag
*/
2011-02-09 17:31:16 +13:00
function OpenBlock_Handle_Base_tag ( & $res ) {
if ( $res [ 'ArgumentCount' ] != 0 ) throw new SSTemplateParseException ( 'Base_tag takes no arguments' , $this );
return '$val .= SSViewer::get_base_tag($val);' ;
}
2011-02-15 10:03:42 +13:00
/**
* This is an open block handler , for the <% current_page %> tag
*/
2011-02-09 17:31:16 +13:00
function OpenBlock_Handle_Current_page ( & $res ) {
if ( $res [ 'ArgumentCount' ] != 0 ) throw new SSTemplateParseException ( 'Current_page takes no arguments' , $this );
return '$val .= $_SERVER[SCRIPT_URL];' ;
}
/*!*
2011-02-15 10:03:42 +13:00
# This is used to detect when we have a mismatched closing tag (i.e., one with no equivilent opening tag)
# Because of parser limitations, this can only be used at the top nesting level of a template. Other mismatched
# closing tags are detected as an invalid open tag
2011-02-28 16:12:41 +13:00
MismatchedEndBlock : '<%' < 'end_' : Word > '%>'
2011-02-09 17:31:16 +13:00
*/
function MismatchedEndBlock__finalise ( & $res ) {
$blockname = $res [ 'Word' ][ 'text' ];
throw new SSTemplateParseException ( 'Unexpected close tag end_' . $blockname . ' encountered. Perhaps you have mis-nested blocks, or have mis-spelled a tag?' , $this );
}
/*!*
2011-02-15 10:03:42 +13:00
# This is used to detect a malformed opening tag - one where the tag is opened with the "<%" characters, but
# the tag is not structured properly
2011-02-09 17:31:16 +13:00
MalformedOpenTag : '<%' < ! NotBlockTag Tag : Word ! ( ( [ : BlockArguments ] ) ? > '%>' )
*/
function MalformedOpenTag__finalise ( & $res ) {
$tag = $res [ 'Tag' ][ 'text' ];
throw new SSTemplateParseException ( " Malformed opening block tag $tag . Perhaps you have tried to use operators? " , $this );
}
/*!*
2011-02-15 10:03:42 +13:00
# This is used to detect a malformed end tag - one where the tag is opened with the "<%" characters, but
# the tag is not structured properly
2011-02-09 17:31:16 +13:00
MalformedCloseTag : '<%' < Tag : ( 'end_' : Word ) ! ( > '%>' )
*/
function MalformedCloseTag__finalise ( & $res ) {
$tag = $res [ 'Tag' ][ 'text' ];
throw new SSTemplateParseException ( " Malformed closing block tag $tag . Perhaps you have tried to pass an argument to one? " , $this );
}
/*!*
2011-02-15 10:03:42 +13:00
# This is used to detect a malformed tag. It's mostly to keep the Template match rule a bit shorter
2011-02-09 17:31:16 +13:00
MalformedBlock : MalformedOpenTag | MalformedCloseTag
*/
/*!*
2011-02-15 10:03:42 +13:00
# This is used to remove template comments
2011-02-09 17:31:16 +13:00
Comment : " <%-- " ( ! " --%> " /./ ) + " --%> "
*/
function Comment__construct ( & $res ) {
$res [ 'php' ] = '' ;
}
2011-02-28 16:12:41 +13:00
/*!*
2011-02-09 17:31:16 +13:00
2011-02-28 16:12:41 +13:00
# TopTemplate is the same as Template, but should only be used at the top level (not nested), as it includes
# MismatchedEndBlock detection, which only works at the top level
TopTemplate extends Template ( TemplateMatcher = Template ); MalformedBlock => MalformedBlock | MismatchedEndBlock
*/
/**
* The TopTemplate also includes the opening stanza to start off the template
*/
function TopTemplate__construct ( & $res ) {
$res [ 'php' ] = " <?php " . PHP_EOL ;
}
2011-02-09 17:31:16 +13:00
/*!*
2011-02-15 10:03:42 +13:00
# Text matches anything that isn't a template command (not an injection, block of any kind or comment)
2011-03-17 13:08:13 +13:00
Text : (
# Any set of characters that aren't potentially a control mark or an escaped character
/ [ ^< $ { \\ ] + / |
# An escaped character
/ ( \\ . ) / |
# A '<' that isn't the start of a block tag
'<' ! '%' |
# A '$' that isn't the start of an injection
'$' ! ( / [ A - Za - z_ ] / ) |
# A '{' that isn't the start of an injection
'{' ! '$' |
# A '{$' that isn't the start of an injection
'{$' ! ( / [ A - Za - z_ ] / )
2011-02-09 17:31:16 +13:00
) +
*/
2011-02-28 16:12:41 +13:00
/**
* We convert text
*/
function Text__finalise ( & $res ) {
$text = $res [ 'text' ];
2011-03-17 13:08:13 +13:00
// Unescape any escaped characters in the text, then put back escapes for any single quotes and backslashes
$text = stripslashes ( $text );
$text = addcslashes ( $text , '\'\\' );
// TODO: This is pretty ugly & gets applied on all files not just html. I wonder if we can make this non-dynamically calculated
2011-02-09 17:31:16 +13:00
$text = preg_replace (
'/href\s*\=\s*\"\#/' ,
2011-10-18 11:42:55 +02:00
'href="\' . (SSViewer::$options[\'rewriteHashlinks\'] ? strip_tags( $_SERVER[\'REQUEST_URI\'] ) : "") . \'#' ,
2011-02-09 17:31:16 +13:00
$text
);
2011-03-17 13:08:13 +13:00
$res [ 'php' ] .= '$val .= \'' . $text . '\';' . PHP_EOL ;
2011-02-09 17:31:16 +13:00
}
2011-02-28 16:12:41 +13:00
2011-02-15 10:03:42 +13:00
/******************
* Here ends the parser itself . Below are utility methods to use the parser
*/
/**
* Compiles some passed template source code into the php code that will execute as per the template source .
*
* @ static
* @ throws SSTemplateParseException
* @ param $string - The source of the template
* @ param string $templateName - The name of the template , normally the filename the template source was loaded from
* @ param bool $includeDebuggingComments - True is debugging comments should be included in the output
* @ return mixed | string - The php that , when executed ( via include or exec ) will behave as per the template source
*/
2011-02-09 17:31:16 +13:00
static function compileString ( $string , $templateName = " " , $includeDebuggingComments = false ) {
2011-03-25 12:28:03 +13:00
if ( ! trim ( $string )) {
$code = '' ;
}
else {
// Construct a parser instance
$parser = new SSTemplateParser ( $string );
$parser -> includeDebuggingComments = $includeDebuggingComments ;
// Ignore UTF8 BOM at begining of string. TODO: Confirm this is needed, make sure SSViewer handles UTF (and other encodings) properly
if ( substr ( $string , 0 , 3 ) == pack ( " CCC " , 0xef , 0xbb , 0xbf )) $parser -> pos = 3 ;
// Match the source against the parser
$result = $parser -> match_TopTemplate ();
if ( ! $result ) throw new SSTemplateParseException ( 'Unexpected problem parsing template' , $parser );
// Get the result
$code = $result [ 'php' ];
}
2011-02-09 17:31:16 +13:00
2011-02-15 10:03:42 +13:00
// Include top level debugging comments if desired
2011-02-09 17:31:16 +13:00
if ( $includeDebuggingComments && $templateName && stripos ( $code , " <?xml " ) === false ) {
// If this template is a full HTML page, then put the comments just inside the HTML tag to prevent any IE glitches
if ( stripos ( $code , " <html " ) !== false ) {
$code = preg_replace ( '/(<html[^>]*>)/i' , " \\ 1<!-- template $templateName --> " , $code );
$code = preg_replace ( '/(<\/html[^>]*>)/i' , " <!-- end template $templateName --> \\ 1 " , $code );
} else {
2011-12-17 17:35:26 +13:00
$code = str_replace ( '<?php' . PHP_EOL , '<?php' . PHP_EOL . '$val .= \'<!-- template ' . $templateName . ' -->\';' . " \n " , $code );
$code .= " \n " . '$val .= \'<!-- end template ' . $templateName . ' -->\';' ;
2011-02-09 17:31:16 +13:00
}
}
return $code ;
}
2011-02-15 10:03:42 +13:00
/**
* Compiles some file that contains template source code , and returns the php code that will execute as per that
* source
*
* @ static
* @ param $template - A file path that contains template source code
* @ return mixed | string - The php that , when executed ( via include or exec ) will behave as per the template source
*/
2011-02-09 17:31:16 +13:00
static function compileFile ( $template ) {
2011-02-15 10:03:42 +13:00
return self :: compileString ( file_get_contents ( $template ), $template );
2011-02-09 17:31:16 +13:00
}
}