2017-12-30 14:31:07 -08:00
|
|
|
file (File):
|
|
|
|
{| shebang?
|
|
|
|
(ignored_line %nl)*
|
2018-01-07 16:47:14 -08:00
|
|
|
statement (nodent statement)*
|
2017-12-30 14:31:07 -08:00
|
|
|
(%nl ignored_line)*
|
|
|
|
(!. / (("" -> "Parse error") => error)?) |}
|
|
|
|
|
|
|
|
shebang: "#!" [^%nl]* %nl
|
|
|
|
|
|
|
|
statement: functioncall / expression
|
|
|
|
|
2018-01-08 18:53:57 -08:00
|
|
|
indented_block (Block):
|
2018-01-30 16:40:55 -08:00
|
|
|
{| (":" / "(..)")? indent
|
2018-01-07 16:47:14 -08:00
|
|
|
statement (nodent statement)*
|
2018-01-08 18:53:57 -08:00
|
|
|
(dedent / (("" -> "Error while parsing block") => error))
|
2017-12-30 14:31:07 -08:00
|
|
|
|}
|
|
|
|
|
|
|
|
inline_nomsu (Nomsu): "\" inline_expression
|
|
|
|
indented_nomsu (Nomsu): "\" expression
|
|
|
|
|
|
|
|
inline_expression:
|
2018-01-11 01:09:26 -08:00
|
|
|
number / variable / inline_text / inline_list / inline_dict / inline_nomsu
|
2018-01-19 17:29:44 -08:00
|
|
|
/ ("(" %ws* (inline_functioncall / inline_expression) %ws* ")")
|
|
|
|
indented_expression:
|
2018-01-11 01:09:26 -08:00
|
|
|
indented_text / indented_nomsu / indented_list / indented_dict / indented_block
|
2018-01-19 17:29:44 -08:00
|
|
|
expression:
|
|
|
|
inline_expression / (":" %ws* (inline_functioncall / inline_expression)) / indented_expression
|
2017-12-30 14:31:07 -08:00
|
|
|
|
|
|
|
-- Function calls need at least one word in them
|
|
|
|
inline_functioncall (FunctionCall):
|
2018-01-19 17:29:44 -08:00
|
|
|
{| (inline_expression %ws*)* word (%ws* (inline_expression / word))*
|
|
|
|
(%ws* ":" %ws* (inline_functioncall / inline_expression))?|}
|
2017-12-30 14:31:07 -08:00
|
|
|
functioncall (FunctionCall):
|
|
|
|
{| (expression (dotdot / %ws*))* word ((dotdot / %ws*) (expression / word))* |}
|
|
|
|
|
2018-01-03 19:26:41 -08:00
|
|
|
word (Word): { %operator / (!number plain_word) }
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-01-11 01:09:26 -08:00
|
|
|
inline_text (Text):
|
2018-01-30 15:10:37 -08:00
|
|
|
!('".."' eol)
|
2017-12-30 14:31:07 -08:00
|
|
|
'"' {|
|
2018-01-26 14:53:02 -08:00
|
|
|
({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~}
|
2018-01-30 15:13:04 -08:00
|
|
|
/ inline_text_interpolation)*
|
2017-12-30 14:31:07 -08:00
|
|
|
|} '"'
|
2018-01-30 15:10:37 -08:00
|
|
|
-- Have to use "%indent" instead of "indent" etc. to avoid messing up text lines that start with "#"
|
2018-01-11 01:09:26 -08:00
|
|
|
indented_text (Text):
|
2018-01-30 15:10:37 -08:00
|
|
|
'".."' eol %nl {|
|
|
|
|
{~ (%nl*) (%indent -> "") ~}
|
2018-01-03 00:52:01 -08:00
|
|
|
({~
|
2018-01-30 15:10:37 -08:00
|
|
|
(("\\" -> "\") / (("\" eol %nl+ %nodent "..") -> "")
|
|
|
|
/ (%nl+ {~ %nodent -> "" ~}) / [^%nl\])+
|
2018-01-11 01:09:26 -08:00
|
|
|
~} / text_interpolation)*
|
2018-01-30 15:10:37 -08:00
|
|
|
|} (((!.) &%dedent) / (&(%nl %dedent)) / (("" -> "Error while parsing Text") => error))
|
2018-01-30 15:13:04 -08:00
|
|
|
inline_text_interpolation:
|
2018-01-30 15:10:37 -08:00
|
|
|
"\" (
|
|
|
|
variable / inline_list / inline_dict / inline_text
|
|
|
|
/ ("(" %ws* (inline_functioncall / inline_expression) %ws* ")")
|
2018-01-30 15:13:04 -08:00
|
|
|
)
|
|
|
|
text_interpolation:
|
|
|
|
inline_text_interpolation /
|
|
|
|
("\" (
|
|
|
|
variable / inline_list / inline_dict / inline_text
|
|
|
|
/ ("(" %ws* (inline_functioncall / inline_expression) %ws* ")")
|
2018-01-30 15:10:37 -08:00
|
|
|
/ (%ws* (block_comment / line_comment)? nodent "..")
|
|
|
|
/ (indented_text %nl+ %nodent "..")
|
|
|
|
/ ((indented_list / indented_block) nodent "..")
|
2018-01-30 15:13:04 -08:00
|
|
|
))
|
2017-12-30 14:31:07 -08:00
|
|
|
|
|
|
|
number (Number): (("-"? (([0-9]+ "." [0-9]+) / ("." [0-9]+) / ([0-9]+)))-> tonumber)
|
|
|
|
|
|
|
|
-- Variables can be nameless (i.e. just %) and can't contain operators like apostrophe
|
|
|
|
-- which is a hack to allow %'s to parse as "%" and "' s" separately
|
|
|
|
variable (Var): "%" { plain_word? }
|
|
|
|
|
|
|
|
inline_list (List):
|
2018-01-19 17:29:44 -08:00
|
|
|
!('[..]')
|
2017-12-30 14:31:07 -08:00
|
|
|
"[" %ws* {| (inline_list_item (comma inline_list_item)* comma?)? |} %ws* "]"
|
|
|
|
indented_list (List):
|
|
|
|
"[..]" indent {|
|
|
|
|
list_line (nodent list_line)*
|
|
|
|
|}
|
|
|
|
(dedent / (("" -> "Error while parsing list") => error))
|
|
|
|
list_line:
|
2018-01-19 17:29:44 -08:00
|
|
|
((functioncall / expression) !comma)
|
|
|
|
/ (inline_list_item (comma list_line?)?)
|
2017-12-30 14:31:07 -08:00
|
|
|
inline_list_item: inline_functioncall / inline_expression
|
|
|
|
|
2018-01-03 00:52:01 -08:00
|
|
|
inline_dict (Dict):
|
2018-01-19 17:29:44 -08:00
|
|
|
!('{..}')
|
2018-01-03 00:52:01 -08:00
|
|
|
"{" %ws* {| (inline_dict_item (comma inline_dict_item)* comma?)? |} %ws* "}"
|
|
|
|
indented_dict (Dict):
|
|
|
|
"{..}" indent {|
|
|
|
|
dict_line (nodent dict_line)*
|
|
|
|
|}
|
|
|
|
(dedent / (("" -> "Error while parsing dict") => error))
|
|
|
|
dict_line:
|
2018-01-19 17:29:44 -08:00
|
|
|
({| {:dict_key: inline_expression / word :} %ws* ":" %ws* {:dict_value: functioncall / expression :} |} !comma)
|
|
|
|
/ (inline_dict_item (comma dict_line?)?)
|
2018-01-03 00:52:01 -08:00
|
|
|
inline_dict_item:
|
2018-01-19 17:29:44 -08:00
|
|
|
{| {:dict_key: inline_expression / word :} %ws* ":" %ws* {:dict_value: inline_functioncall / inline_expression :} |}
|
2018-01-03 00:52:01 -08:00
|
|
|
|
2018-01-30 15:10:37 -08:00
|
|
|
block_comment(Comment): "#.." { [^%nl]* (%nl+ %indent [^%nl]* (%nl+ %nodent [^%nl]*)* %dedent)? }
|
2018-01-15 15:50:18 -08:00
|
|
|
line_comment(Comment): "#" { [^%nl]* }
|
2017-12-30 14:31:07 -08:00
|
|
|
|
|
|
|
eol: %ws* line_comment? (!. / &%nl)
|
2018-01-19 17:29:44 -08:00
|
|
|
ignored_line: (%nodent (block_comment / line_comment)) / (%ws* (!. / &%nl))
|
|
|
|
indent: eol (%nl ignored_line)* %nl %indent ((block_comment/line_comment) (%nl ignored_line)* nodent)?
|
|
|
|
nodent: eol (%nl ignored_line)* %nl %nodent
|
|
|
|
dedent: eol (%nl ignored_line)* (((!.) &%dedent) / (&(%nl %dedent)))
|
2017-12-30 14:31:07 -08:00
|
|
|
comma: %ws* "," %ws*
|
|
|
|
dotdot: nodent ".." %ws*
|
|
|
|
plain_word: ([a-zA-Z0-9_] / %utf8_char)+
|