parse.y 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862
  1. // BUILD file parser.
  2. // This is a yacc grammar. Its lexer is in lex.go.
  3. //
  4. // For a good introduction to writing yacc grammars, see
  5. // Kernighan and Pike's book The Unix Programming Environment.
  6. //
  7. // The definitive yacc manual is
  8. // Stephen C. Johnson and Ravi Sethi, "Yacc: A Parser Generator",
  9. // online at http://plan9.bell-labs.com/sys/doc/yacc.pdf.
  10. %{
  11. package build
  12. %}
  13. // The generated parser puts these fields in a struct named yySymType.
  14. // (The name %union is historical, but it is inaccurate for Go.)
  15. %union {
  16. // input tokens
  17. tok string // raw input syntax
  18. str string // decoding of quoted string
  19. pos Position // position of token
  20. triple bool // was string triple quoted?
  21. // partial syntax trees
  22. expr Expr
  23. exprs []Expr
  24. forc *ForClause
  25. ifs []*IfClause
  26. forifs *ForClauseWithIfClausesOpt
  27. forsifs []*ForClauseWithIfClausesOpt
  28. string *StringExpr
  29. strings []*StringExpr
  30. block CodeBlock
  31. // supporting information
  32. comma Position // position of trailing comma in list, if present
  33. lastRule Expr // most recent rule, to attach line comments to
  34. }
  35. // These declarations set the type for a $ reference ($$, $1, $2, ...)
  36. // based on the kind of symbol it refers to. Other fields can be referred
  37. // to explicitly, as in $<tok>1.
  38. //
  39. // %token is for input tokens generated by the lexer.
  40. // %type is for higher-level grammar rules defined here.
  41. //
  42. // It is possible to put multiple tokens per line, but it is easier to
  43. // keep ordered using a sparser one-per-line list.
  44. %token <pos> '%'
  45. %token <pos> '('
  46. %token <pos> ')'
  47. %token <pos> '*'
  48. %token <pos> '+'
  49. %token <pos> ','
  50. %token <pos> '-'
  51. %token <pos> '.'
  52. %token <pos> '/'
  53. %token <pos> ':'
  54. %token <pos> '<'
  55. %token <pos> '='
  56. %token <pos> '>'
  57. %token <pos> '['
  58. %token <pos> ']'
  59. %token <pos> '{'
  60. %token <pos> '}'
  61. // By convention, yacc token names are all caps.
  62. // However, we do not want to export them from the Go package
  63. // we are creating, so prefix them all with underscores.
  64. %token <pos> _AUGM // augmented assignment
  65. %token <pos> _AND // keyword and
  66. %token <pos> _COMMENT // top-level # comment
  67. %token <pos> _EOF // end of file
  68. %token <pos> _EQ // operator ==
  69. %token <pos> _FOR // keyword for
  70. %token <pos> _GE // operator >=
  71. %token <pos> _IDENT // non-keyword identifier or number
  72. %token <pos> _IF // keyword if
  73. %token <pos> _ELSE // keyword else
  74. %token <pos> _ELIF // keyword elif
  75. %token <pos> _IN // keyword in
  76. %token <pos> _IS // keyword is
  77. %token <pos> _LAMBDA // keyword lambda
  78. %token <pos> _LOAD // keyword load
  79. %token <pos> _LE // operator <=
  80. %token <pos> _NE // operator !=
  81. %token <pos> _NOT // keyword not
  82. %token <pos> _OR // keyword or
  83. %token <pos> _PYTHON // uninterpreted Python block
  84. %token <pos> _STRING // quoted string
  85. %token <pos> _DEF // keyword def
  86. %token <pos> _RETURN // keyword return
  87. %token <pos> _INDENT // indentation
  88. %token <pos> _UNINDENT // unindentation
  89. %type <pos> comma_opt
  90. %type <expr> expr
  91. %type <expr> expr_opt
  92. %type <expr> primary_expr
  93. %type <exprs> exprs
  94. %type <exprs> exprs_opt
  95. %type <exprs> primary_exprs
  96. %type <forc> for_clause
  97. %type <forifs> for_clause_with_if_clauses_opt
  98. %type <forsifs> for_clauses_with_if_clauses_opt
  99. %type <expr> ident
  100. %type <ifs> if_clauses_opt
  101. %type <exprs> stmts
  102. %type <exprs> stmt // a simple_stmt or a for/if/def block
  103. %type <expr> block_stmt // a single for/if/def statement
  104. %type <expr> if_else_block // a single if-else statement
  105. %type <exprs> simple_stmt // One or many small_stmts on one line, e.g. 'a = f(x); return str(a)'
  106. %type <expr> small_stmt // A single statement, e.g. 'a = f(x)'
  107. %type <exprs> small_stmts_continuation // A sequence of `';' small_stmt`
  108. %type <expr> keyvalue
  109. %type <exprs> keyvalues
  110. %type <exprs> keyvalues_no_comma
  111. %type <string> string
  112. %type <strings> strings
  113. %type <block> suite
  114. // Operator precedence.
  115. // Operators listed lower in the table bind tighter.
  116. // We tag rules with this fake, low precedence to indicate
  117. // that when the rule is involved in a shift/reduce
  118. // conflict, we prefer that the parser shift (try for a longer parse).
  119. // Shifting is the default resolution anyway, but stating it explicitly
  120. // silences yacc's warning for that specific case.
  121. %left ShiftInstead
  122. %left '\n'
  123. %left _ASSERT
  124. // '=' and augmented assignments have the lowest precedence
  125. // e.g. "x = a if c > 0 else 'bar'"
  126. // followed by
  127. // 'if' and 'else' which have lower precedence than all other operators.
  128. // e.g. "a, b if c > 0 else 'foo'" is either a tuple of (a,b) or 'foo'
  129. // and not a tuple of "(a, (b if ... ))"
  130. %left '=' _AUGM
  131. %left _IF _ELSE _ELIF
  132. %left ','
  133. %left ':'
  134. %left _IN _NOT _IS
  135. %left _OR
  136. %left _AND
  137. %left '<' '>' _EQ _NE _LE _GE
  138. %left '+' '-'
  139. %left '*' '/' '%'
  140. %left '.' '[' '('
  141. %right _UNARY
  142. %left _STRING
  143. %%
  144. // Grammar rules.
  145. //
  146. // A note on names: if foo is a rule, then foos is a sequence of foos
  147. // (with interleaved commas or other syntax as appropriate)
  148. // and foo_opt is an optional foo.
  149. file:
  150. stmts _EOF
  151. {
  152. yylex.(*input).file = &File{Stmt: $1}
  153. return 0
  154. }
  155. suite:
  156. '\n' _INDENT stmts _UNINDENT
  157. {
  158. $$ = CodeBlock{
  159. Start: $2,
  160. Statements: $3,
  161. End: End{Pos: $4},
  162. }
  163. }
  164. | simple_stmt
  165. {
  166. // simple_stmt is never empty
  167. start, _ := $1[0].Span()
  168. _, end := $1[len($1)-1].Span()
  169. $$ = CodeBlock{
  170. Start: start,
  171. Statements: $1,
  172. End: End{Pos: end},
  173. }
  174. }
  175. stmts:
  176. {
  177. $$ = nil
  178. $<lastRule>$ = nil
  179. }
  180. | stmts stmt
  181. {
  182. // If this statement follows a comment block,
  183. // attach the comments to the statement.
  184. if cb, ok := $<lastRule>1.(*CommentBlock); ok {
  185. $$ = append($1[:len($1)-1], $2...)
  186. $2[0].Comment().Before = cb.After
  187. $<lastRule>$ = $2[len($2)-1]
  188. break
  189. }
  190. // Otherwise add to list.
  191. $$ = append($1, $2...)
  192. $<lastRule>$ = $2[len($2)-1]
  193. // Consider this input:
  194. //
  195. // foo()
  196. // # bar
  197. // baz()
  198. //
  199. // If we've just parsed baz(), the # bar is attached to
  200. // foo() as an After comment. Make it a Before comment
  201. // for baz() instead.
  202. if x := $<lastRule>1; x != nil {
  203. com := x.Comment()
  204. // stmt is never empty
  205. $2[0].Comment().Before = com.After
  206. com.After = nil
  207. }
  208. }
  209. | stmts '\n'
  210. {
  211. // Blank line; sever last rule from future comments.
  212. $$ = $1
  213. $<lastRule>$ = nil
  214. }
  215. | stmts _COMMENT '\n'
  216. {
  217. $$ = $1
  218. $<lastRule>$ = $<lastRule>1
  219. if $<lastRule>$ == nil {
  220. cb := &CommentBlock{Start: $2}
  221. $$ = append($$, cb)
  222. $<lastRule>$ = cb
  223. }
  224. com := $<lastRule>$.Comment()
  225. com.After = append(com.After, Comment{Start: $2, Token: $<tok>2})
  226. }
  227. stmt:
  228. simple_stmt
  229. {
  230. $$ = $1
  231. }
  232. | block_stmt
  233. {
  234. $$ = []Expr{$1}
  235. }
  236. block_stmt:
  237. _DEF _IDENT '(' exprs_opt ')' ':' suite
  238. {
  239. $$ = &FuncDef{
  240. Start: $1,
  241. Name: $<tok>2,
  242. ListStart: $3,
  243. Args: $4,
  244. Body: $7,
  245. End: $7.End,
  246. ForceCompact: forceCompact($3, $4, $5),
  247. ForceMultiLine: forceMultiLine($3, $4, $5),
  248. }
  249. }
  250. | _FOR primary_exprs _IN expr ':' suite
  251. {
  252. $$ = &ForLoop{
  253. Start: $1,
  254. LoopVars: $2,
  255. Iterable: $4,
  256. Body: $6,
  257. End: $6.End,
  258. }
  259. }
  260. | if_else_block
  261. {
  262. $$ = $1
  263. }
  264. if_else_block:
  265. _IF expr ':' suite
  266. {
  267. $$ = &IfElse{
  268. Start: $1,
  269. Conditions: []Condition{
  270. Condition{
  271. If: $2,
  272. Then: $4,
  273. },
  274. },
  275. End: $4.End,
  276. }
  277. }
  278. | if_else_block elif expr ':' suite
  279. {
  280. block := $1.(*IfElse)
  281. block.Conditions = append(block.Conditions, Condition{
  282. If: $3,
  283. Then: $5,
  284. })
  285. block.End = $5.End
  286. $$ = block
  287. }
  288. | if_else_block _ELSE ':' suite
  289. {
  290. block := $1.(*IfElse)
  291. block.Conditions = append(block.Conditions, Condition{
  292. Then: $4,
  293. })
  294. block.End = $4.End
  295. $$ = block
  296. }
  297. elif:
  298. _ELSE _IF
  299. | _ELIF
  300. simple_stmt:
  301. small_stmt small_stmts_continuation semi_opt '\n'
  302. {
  303. $$ = append([]Expr{$1}, $2...)
  304. $<lastRule>$ = $$[len($$)-1]
  305. }
  306. small_stmts_continuation:
  307. {
  308. $$ = []Expr{}
  309. }
  310. | small_stmts_continuation ';' small_stmt
  311. {
  312. $$ = append($1, $3)
  313. }
  314. small_stmt:
  315. expr %prec ShiftInstead
  316. | _RETURN expr
  317. {
  318. _, end := $2.Span()
  319. $$ = &ReturnExpr{
  320. X: $2,
  321. End: end,
  322. }
  323. }
  324. | _RETURN
  325. {
  326. $$ = &ReturnExpr{End: $1}
  327. }
  328. | _PYTHON
  329. {
  330. $$ = &PythonBlock{Start: $1, Token: $<tok>1}
  331. }
  332. semi_opt:
  333. | ';'
  334. primary_expr:
  335. ident
  336. | primary_expr '.' _IDENT
  337. {
  338. $$ = &DotExpr{
  339. X: $1,
  340. Dot: $2,
  341. NamePos: $3,
  342. Name: $<tok>3,
  343. }
  344. }
  345. | _LOAD '(' exprs_opt ')'
  346. {
  347. $$ = &CallExpr{
  348. X: &LiteralExpr{Start: $1, Token: "load"},
  349. ListStart: $2,
  350. List: $3,
  351. End: End{Pos: $4},
  352. ForceCompact: forceCompact($2, $3, $4),
  353. ForceMultiLine: forceMultiLine($2, $3, $4),
  354. }
  355. }
  356. | primary_expr '(' exprs_opt ')'
  357. {
  358. $$ = &CallExpr{
  359. X: $1,
  360. ListStart: $2,
  361. List: $3,
  362. End: End{Pos: $4},
  363. ForceCompact: forceCompact($2, $3, $4),
  364. ForceMultiLine: forceMultiLine($2, $3, $4),
  365. }
  366. }
  367. | primary_expr '[' expr ']'
  368. {
  369. $$ = &IndexExpr{
  370. X: $1,
  371. IndexStart: $2,
  372. Y: $3,
  373. End: $4,
  374. }
  375. }
  376. | primary_expr '[' expr_opt ':' expr_opt ']'
  377. {
  378. $$ = &SliceExpr{
  379. X: $1,
  380. SliceStart: $2,
  381. From: $3,
  382. FirstColon: $4,
  383. To: $5,
  384. End: $6,
  385. }
  386. }
  387. | primary_expr '[' expr_opt ':' expr_opt ':' expr_opt ']'
  388. {
  389. $$ = &SliceExpr{
  390. X: $1,
  391. SliceStart: $2,
  392. From: $3,
  393. FirstColon: $4,
  394. To: $5,
  395. SecondColon: $6,
  396. Step: $7,
  397. End: $8,
  398. }
  399. }
  400. | primary_expr '(' expr for_clauses_with_if_clauses_opt ')'
  401. {
  402. $$ = &CallExpr{
  403. X: $1,
  404. ListStart: $2,
  405. List: []Expr{
  406. &ListForExpr{
  407. Brack: "",
  408. Start: $2,
  409. X: $3,
  410. For: $4,
  411. End: End{Pos: $5},
  412. },
  413. },
  414. End: End{Pos: $5},
  415. }
  416. }
  417. | strings %prec ShiftInstead
  418. {
  419. if len($1) == 1 {
  420. $$ = $1[0]
  421. break
  422. }
  423. $$ = $1[0]
  424. for _, x := range $1[1:] {
  425. _, end := $$.Span()
  426. $$ = binary($$, end, "+", x)
  427. }
  428. }
  429. | '[' exprs_opt ']'
  430. {
  431. $$ = &ListExpr{
  432. Start: $1,
  433. List: $2,
  434. Comma: $<comma>2,
  435. End: End{Pos: $3},
  436. ForceMultiLine: forceMultiLine($1, $2, $3),
  437. }
  438. }
  439. | '[' expr for_clauses_with_if_clauses_opt ']'
  440. {
  441. exprStart, _ := $2.Span()
  442. $$ = &ListForExpr{
  443. Brack: "[]",
  444. Start: $1,
  445. X: $2,
  446. For: $3,
  447. End: End{Pos: $4},
  448. ForceMultiLine: $1.Line != exprStart.Line,
  449. }
  450. }
  451. | '(' expr for_clauses_with_if_clauses_opt ')'
  452. {
  453. exprStart, _ := $2.Span()
  454. $$ = &ListForExpr{
  455. Brack: "()",
  456. Start: $1,
  457. X: $2,
  458. For: $3,
  459. End: End{Pos: $4},
  460. ForceMultiLine: $1.Line != exprStart.Line,
  461. }
  462. }
  463. | '{' keyvalue for_clauses_with_if_clauses_opt '}'
  464. {
  465. exprStart, _ := $2.Span()
  466. $$ = &ListForExpr{
  467. Brack: "{}",
  468. Start: $1,
  469. X: $2,
  470. For: $3,
  471. End: End{Pos: $4},
  472. ForceMultiLine: $1.Line != exprStart.Line,
  473. }
  474. }
  475. | '{' keyvalues '}'
  476. {
  477. $$ = &DictExpr{
  478. Start: $1,
  479. List: $2,
  480. Comma: $<comma>2,
  481. End: End{Pos: $3},
  482. ForceMultiLine: forceMultiLine($1, $2, $3),
  483. }
  484. }
  485. | '{' exprs_opt '}'
  486. {
  487. $$ = &SetExpr{
  488. Start: $1,
  489. List: $2,
  490. Comma: $<comma>2,
  491. End: End{Pos: $3},
  492. ForceMultiLine: forceMultiLine($1, $2, $3),
  493. }
  494. }
  495. | '(' exprs_opt ')'
  496. {
  497. if len($2) == 1 && $<comma>2.Line == 0 {
  498. // Just a parenthesized expression, not a tuple.
  499. $$ = &ParenExpr{
  500. Start: $1,
  501. X: $2[0],
  502. End: End{Pos: $3},
  503. ForceMultiLine: forceMultiLine($1, $2, $3),
  504. }
  505. } else {
  506. $$ = &TupleExpr{
  507. Start: $1,
  508. List: $2,
  509. Comma: $<comma>2,
  510. End: End{Pos: $3},
  511. ForceCompact: forceCompact($1, $2, $3),
  512. ForceMultiLine: forceMultiLine($1, $2, $3),
  513. }
  514. }
  515. }
  516. | '-' primary_expr %prec _UNARY { $$ = unary($1, $<tok>1, $2) }
  517. expr:
  518. primary_expr
  519. | _LAMBDA exprs ':' expr
  520. {
  521. $$ = &LambdaExpr{
  522. Lambda: $1,
  523. Var: $2,
  524. Colon: $3,
  525. Expr: $4,
  526. }
  527. }
  528. | _NOT expr %prec _UNARY { $$ = unary($1, $<tok>1, $2) }
  529. | '*' expr %prec _UNARY { $$ = unary($1, $<tok>1, $2) }
  530. | expr '*' expr { $$ = binary($1, $2, $<tok>2, $3) }
  531. | expr '%' expr { $$ = binary($1, $2, $<tok>2, $3) }
  532. | expr '/' expr { $$ = binary($1, $2, $<tok>2, $3) }
  533. | expr '+' expr { $$ = binary($1, $2, $<tok>2, $3) }
  534. | expr '-' expr { $$ = binary($1, $2, $<tok>2, $3) }
  535. | expr '<' expr { $$ = binary($1, $2, $<tok>2, $3) }
  536. | expr '>' expr { $$ = binary($1, $2, $<tok>2, $3) }
  537. | expr _EQ expr { $$ = binary($1, $2, $<tok>2, $3) }
  538. | expr _LE expr { $$ = binary($1, $2, $<tok>2, $3) }
  539. | expr _NE expr { $$ = binary($1, $2, $<tok>2, $3) }
  540. | expr _GE expr { $$ = binary($1, $2, $<tok>2, $3) }
  541. | expr '=' expr { $$ = binary($1, $2, $<tok>2, $3) }
  542. | expr _AUGM expr { $$ = binary($1, $2, $<tok>2, $3) }
  543. | expr _IN expr { $$ = binary($1, $2, $<tok>2, $3) }
  544. | expr _NOT _IN expr { $$ = binary($1, $2, "not in", $4) }
  545. | expr _OR expr { $$ = binary($1, $2, $<tok>2, $3) }
  546. | expr _AND expr { $$ = binary($1, $2, $<tok>2, $3) }
  547. | expr _IS expr
  548. {
  549. if b, ok := $3.(*UnaryExpr); ok && b.Op == "not" {
  550. $$ = binary($1, $2, "is not", b.X)
  551. } else {
  552. $$ = binary($1, $2, $<tok>2, $3)
  553. }
  554. }
  555. | expr _IF expr _ELSE expr
  556. {
  557. $$ = &ConditionalExpr{
  558. Then: $1,
  559. IfStart: $2,
  560. Test: $3,
  561. ElseStart: $4,
  562. Else: $5,
  563. }
  564. }
  565. expr_opt:
  566. {
  567. $$ = nil
  568. }
  569. | expr
  570. // comma_opt is an optional comma. If the comma is present,
  571. // the rule's value is the position of the comma. Otherwise
  572. // the rule's value is the zero position. Tracking this
  573. // lets us distinguish (x) and (x,).
  574. comma_opt:
  575. {
  576. $$ = Position{}
  577. }
  578. | ','
  579. keyvalue:
  580. expr ':' expr {
  581. $$ = &KeyValueExpr{
  582. Key: $1,
  583. Colon: $2,
  584. Value: $3,
  585. }
  586. }
  587. keyvalues_no_comma:
  588. keyvalue
  589. {
  590. $$ = []Expr{$1}
  591. }
  592. | keyvalues_no_comma ',' keyvalue
  593. {
  594. $$ = append($1, $3)
  595. }
  596. keyvalues:
  597. keyvalues_no_comma
  598. {
  599. $$ = $1
  600. }
  601. | keyvalues_no_comma ','
  602. {
  603. $$ = $1
  604. }
  605. exprs:
  606. expr
  607. {
  608. $$ = []Expr{$1}
  609. }
  610. | exprs ',' expr
  611. {
  612. $$ = append($1, $3)
  613. }
  614. exprs_opt:
  615. {
  616. $$, $<comma>$ = nil, Position{}
  617. }
  618. | exprs comma_opt
  619. {
  620. $$, $<comma>$ = $1, $2
  621. }
  622. primary_exprs:
  623. primary_expr
  624. {
  625. $$ = []Expr{$1}
  626. }
  627. | primary_exprs ',' primary_expr
  628. {
  629. $$ = append($1, $3)
  630. }
  631. string:
  632. _STRING
  633. {
  634. $$ = &StringExpr{
  635. Start: $1,
  636. Value: $<str>1,
  637. TripleQuote: $<triple>1,
  638. End: $1.add($<tok>1),
  639. Token: $<tok>1,
  640. }
  641. }
  642. strings:
  643. string
  644. {
  645. $$ = []*StringExpr{$1}
  646. }
  647. | strings string
  648. {
  649. $$ = append($1, $2)
  650. }
  651. ident:
  652. _IDENT
  653. {
  654. $$ = &LiteralExpr{Start: $1, Token: $<tok>1}
  655. }
  656. for_clause:
  657. _FOR primary_exprs _IN expr
  658. {
  659. $$ = &ForClause{
  660. For: $1,
  661. Var: $2,
  662. In: $3,
  663. Expr: $4,
  664. }
  665. }
  666. for_clause_with_if_clauses_opt:
  667. for_clause if_clauses_opt {
  668. $$ = &ForClauseWithIfClausesOpt{
  669. For: $1,
  670. Ifs: $2,
  671. }
  672. }
  673. for_clauses_with_if_clauses_opt:
  674. for_clause_with_if_clauses_opt
  675. {
  676. $$ = []*ForClauseWithIfClausesOpt{$1}
  677. }
  678. | for_clauses_with_if_clauses_opt for_clause_with_if_clauses_opt {
  679. $$ = append($1, $2)
  680. }
  681. if_clauses_opt:
  682. {
  683. $$ = nil
  684. }
  685. | if_clauses_opt _IF expr
  686. {
  687. $$ = append($1, &IfClause{
  688. If: $2,
  689. Cond: $3,
  690. })
  691. }
  692. %%
  693. // Go helper code.
  694. // unary returns a unary expression with the given
  695. // position, operator, and subexpression.
  696. func unary(pos Position, op string, x Expr) Expr {
  697. return &UnaryExpr{
  698. OpStart: pos,
  699. Op: op,
  700. X: x,
  701. }
  702. }
  703. // binary returns a binary expression with the given
  704. // operands, position, and operator.
  705. func binary(x Expr, pos Position, op string, y Expr) Expr {
  706. _, xend := x.Span()
  707. ystart, _ := y.Span()
  708. return &BinaryExpr{
  709. X: x,
  710. OpStart: pos,
  711. Op: op,
  712. LineBreak: xend.Line < ystart.Line,
  713. Y: y,
  714. }
  715. }
  716. // isSimpleExpression returns whether an expression is simple and allowed to exist in
  717. // compact forms of sequences.
  718. // The formal criteria are the following: an expression is considered simple if it's
  719. // a literal (variable, string or a number), a literal with a unary operator or an empty sequence.
  720. func isSimpleExpression(expr *Expr) bool {
  721. switch x := (*expr).(type) {
  722. case *LiteralExpr, *StringExpr:
  723. return true
  724. case *UnaryExpr:
  725. _, ok := x.X.(*LiteralExpr)
  726. return ok
  727. case *ListExpr:
  728. return len(x.List) == 0
  729. case *TupleExpr:
  730. return len(x.List) == 0
  731. case *DictExpr:
  732. return len(x.List) == 0
  733. case *SetExpr:
  734. return len(x.List) == 0
  735. default:
  736. return false
  737. }
  738. }
  739. // forceCompact returns the setting for the ForceCompact field for a call or tuple.
  740. //
  741. // NOTE 1: The field is called ForceCompact, not ForceSingleLine,
  742. // because it only affects the formatting associated with the call or tuple syntax,
  743. // not the formatting of the arguments. For example:
  744. //
  745. // call([
  746. // 1,
  747. // 2,
  748. // 3,
  749. // ])
  750. //
  751. // is still a compact call even though it runs on multiple lines.
  752. //
  753. // In contrast the multiline form puts a linebreak after the (.
  754. //
  755. // call(
  756. // [
  757. // 1,
  758. // 2,
  759. // 3,
  760. // ],
  761. // )
  762. //
  763. // NOTE 2: Because of NOTE 1, we cannot use start and end on the
  764. // same line as a signal for compact mode: the formatting of an
  765. // embedded list might move the end to a different line, which would
  766. // then look different on rereading and cause buildifier not to be
  767. // idempotent. Instead, we have to look at properties guaranteed
  768. // to be preserved by the reformatting, namely that the opening
  769. // paren and the first expression are on the same line and that
  770. // each subsequent expression begins on the same line as the last
  771. // one ended (no line breaks after comma).
  772. func forceCompact(start Position, list []Expr, end Position) bool {
  773. if len(list) <= 1 {
  774. // The call or tuple will probably be compact anyway; don't force it.
  775. return false
  776. }
  777. // If there are any named arguments or non-string, non-literal
  778. // arguments, cannot force compact mode.
  779. line := start.Line
  780. for _, x := range list {
  781. start, end := x.Span()
  782. if start.Line != line {
  783. return false
  784. }
  785. line = end.Line
  786. if !isSimpleExpression(&x) {
  787. return false
  788. }
  789. }
  790. return end.Line == line
  791. }
  792. // forceMultiLine returns the setting for the ForceMultiLine field.
  793. func forceMultiLine(start Position, list []Expr, end Position) bool {
  794. if len(list) > 1 {
  795. // The call will be multiline anyway, because it has multiple elements. Don't force it.
  796. return false
  797. }
  798. if len(list) == 0 {
  799. // Empty list: use position of brackets.
  800. return start.Line != end.Line
  801. }
  802. // Single-element list.
  803. // Check whether opening bracket is on different line than beginning of
  804. // element, or closing bracket is on different line than end of element.
  805. elemStart, elemEnd := list[0].Span()
  806. return start.Line != elemStart.Line || end.Line != elemEnd.Line
  807. }