git.delta.rocks / jrsonnet / refs/commits / 0f5424fc1b4b

difftreelog

feat(rowan) alternative object comp syntax

lmvywrutYaroslav Bolyukin2026-05-06parent: #604f09d.patch.diff
in: master

10 files changed

modifiedcrates/jrsonnet-formatter/src/lib.rsdiffbeforeafterboth
--- a/crates/jrsonnet-formatter/src/lib.rs
+++ b/crates/jrsonnet-formatter/src/lib.rs
@@ -13,9 +13,9 @@
 	AstNode, AstToken as _, SyntaxToken,
 	nodes::{
 		Arg, ArgsDesc, Assertion, BinaryOperator, Bind, CompSpec, Destruct, DestructArrayPart,
-		DestructRest, Expr, ExprArray, ExprBase, FieldName, ForSpec, IfSpec, ImportKind, Literal,
-		Member, Name, Number, ObjBody, ObjLocal, ParamsDesc, SliceDesc, SourceFile, Stmt, Suffix,
-		Text, TextKind, UnaryOperator, Visibility,
+		DestructRest, Expr, ExprArray, ExprBase, FieldName, ForObjSpec, ForSpec, IfSpec,
+		ImportKind, Literal, Member, Name, Number, ObjBody, ObjLocal, ParamsDesc, SliceDesc,
+		SourceFile, Stmt, Suffix, Text, TextKind, UnaryOperator, Visibility,
 	},
 };
 
@@ -645,6 +645,11 @@
 		p!(out, str("for ") {self.bind()} str(" in ") {self.expr()});
 	}
 }
+impl Printable for ForObjSpec {
+	fn print(&self, out: &mut PrintItems) {
+		p!(out, str("for [") {self.key()} str("]") {self.visibility()} str(" ") {self.value()} str(" in ") {self.expr()});
+	}
+}
 impl Printable for IfSpec {
 	fn print(&self, out: &mut PrintItems) {
 		p!(out, str("if ") {self.expr()});
@@ -654,6 +659,7 @@
 	fn print(&self, out: &mut PrintItems) {
 		match self {
 			Self::ForSpec(f) => f.print(out),
+			Self::ForObjSpec(f) => f.print(out),
 			Self::IfSpec(i) => i.print(out),
 		}
 	}
modifiedcrates/jrsonnet-rowan-parser/jsonnet.ungramdiffbeforeafterboth
before · crates/jrsonnet-rowan-parser/jsonnet.ungram
1// This file describes structure of jsonnet source code2// It is also used to generate files in src/generated34// Token names ending with `!` are considered meta, and handled specifically56SourceFile = Expr78SuffixIndex =9    '?'?10    '.'11    index:Name12SuffixIndexExpr =13    ('?' '.')?14    '['15    index:Expr16    ']'17SuffixSlice =18    SliceDesc19SuffixApply =20    ArgsDesc21    'tailstrict'?22Suffix =23    SuffixIndex24|   SuffixIndexExpr25|   SuffixSlice26|   SuffixApply2728StmtLocal =29    'local'30    (Bind (',' Bind)* ','?)31    ';'32StmtAssert =33    Assertion34    ';'35Stmt =36    StmtLocal37|   StmtAssert3839ExprBinary =40    lhs:Expr41    BinaryOperator42    rhs:Expr43ExprUnary =44    UnaryOperator45    rhs:Expr46ExprObjExtend =47    lhs:Expr48    rhs:Expr49ExprParened =50    '('51    Expr52    ')'5354ExprLiteral =55    Literal56ExprString =57    Text58ExprNumber =59    Number60ExprArray =61    '['62    (Expr (',' Expr)* ','?)?63    ']'64ExprObject =65    ObjBody66ExprArrayComp =67    '['68    Expr69    ','?70    CompSpec*71    ']'7273ExprImport =74    ImportKind Text7576ImportKind =77    'importstr'78|   'importbin'79|   'import'8081ExprVar =82    name:Name8384ExprIfThenElse =85    'if'86    cond:Expr87    'then'88    then:TrueExpr89    ('else' else_:FalseExpr)?9091ExprFunction =92    'function'93    '('94    ParamsDesc95    ')'96    Expr97ExprError =98    'error'99    Expr100101Expr =102    Stmt*103    ExprBase104    Suffix*105106ExprBase =107    ExprBinary108|   ExprUnary109|   ExprObjExtend110|   ExprParened111|   ExprString112|   ExprNumber113|   ExprLiteral114|   ExprArray115|   ExprObject116|   ExprArrayComp117|   ExprImport118|   ExprVar119|   ExprIfThenElse120|   ExprFunction121|   ExprError122123BinaryOperator =124    '||' | '??' | '&&'125|   '|' | '^' | '&'126|   '==' | '!=' | '<' | '>' | '<=' | '>=' | 'in'127|   '<<' | '>>'128|   '+' | '-'129|   '*' | '/' | '%'130|   'META_OBJECT_APPLY!'131|   'ERROR_NO_OPERATOR!'132133UnaryOperator =134    '-' | '!' | '~'135136SliceDescEnd=Expr137SliceDescStep=Expr138SliceDesc =139    '['140    from:Expr?141    ':'142    (143        end:SliceDescEnd?144        (145            ':'146            step:SliceDescStep?147        )?148    )?149    ']'150151Name =152    'LIT_IDENT!'153154ArgsDesc =155    '('156    (Arg (',' Arg)* ','?)?157    ')'158Arg =159    (name:Name '=')? Expr160161ObjBodyComp =162    '{'163    (MemberComp (',' MemberComp)* ','?)?164    CompSpec*165    '}'166ObjBodyMemberList =167    '{'168    (Member (',' Member)* ','?)?169    '}'170ObjBody =171    ObjBodyComp172|   ObjBodyMemberList173174MemberBindStmt = ObjLocal175MemberAssertStmt = Assertion176MemberFieldNormal =177    FieldName178    '+'?179    Visibility180    Expr181MemberFieldMethod =182    FieldName183    ParamsDesc184    Visibility185    Expr186MemberComp =187    MemberBindStmt188|    MemberFieldNormal189|   MemberFieldMethod190Member =191    MemberBindStmt192|   MemberAssertStmt193|   MemberFieldNormal194|   MemberFieldMethod195196ObjLocal =197    'local'198    Bind199200FieldNameFixed =201    id:Name202|   Text203FieldNameDynamic =204    '['205    Expr206    ']'207FieldName =208    FieldNameFixed209|   FieldNameDynamic210211Visibility =212    ':' ':'? ':'?213214Literal =215    'null'216|   'true'217|   'false'218|   'self'219|   '$'220|   'super'221222Text =223    'LIT_STRING_DOUBLE!'224|   'ERROR_STRING_DOUBLE_UNTERMINATED!'225|   'LIT_STRING_SINGLE!'226|   'ERROR_STRING_SINGLE_UNTERMINATED!'227|   'LIT_STRING_DOUBLE_VERBATIM!'228|   'ERROR_STRING_DOUBLE_VERBATIM_UNTERMINATED!'229|   'LIT_STRING_SINGLE_VERBATIM!'230|   'ERROR_STRING_SINGLE_VERBATIM_UNTERMINATED!'231|   'ERROR_STRING_VERBATIM_MISSING_QUOTES!'232|   'LIT_STRING_BLOCK!'233|   'ERROR_STRING_BLOCK_UNEXPECTED_END!'234|   'ERROR_STRING_BLOCK_MISSING_NEW_LINE!'235|   'ERROR_STRING_BLOCK_MISSING_TERMINATION!'236|   'ERROR_STRING_BLOCK_MISSING_INDENT!'237238Number =239    'LIT_FLOAT!'240|   'ERROR_FLOAT_JUNK_AFTER_POINT!'241|   'ERROR_FLOAT_JUNK_AFTER_EXPONENT!'242|   'ERROR_FLOAT_JUNK_AFTER_EXPONENT_SIGN!'243244ForSpec =245    'for'246    bind:Destruct247    'in'248    Expr249IfSpec =250    'if'251    Expr252CompSpec =253    ForSpec254|   IfSpec255256BindDestruct =257    into:Destruct258    '='259    value:Expr260BindFunction =261    name:Name262    params:ParamsDesc263    '='264    value:Expr265Bind =266    BindDestruct267|   BindFunction268269ParamsDesc =270    '('271    (Param (',' Param)* ','?)?272    ')'273Param =274    Destruct275    (276        '='277        Expr278    )?279280Assertion =281    'assert'282    condition:Expr283    (284        ':'285        message:Expr286    )?287288DestructFull =289    Name290DestructSkip =291    '?'292DestructArray =293    '['294    (295        DestructArrayPart296        (',' DestructArrayPart)*297        ','?298    )?299    ']'300DestructObject =301    '{'302    (303        DestructObjectField304        (',' DestructObjectField)*305        ','?306    )?307    DestructRest?308    ','?309    '}'310Destruct =311    DestructFull312|   DestructSkip313|   DestructArray314|   DestructObject315316DestructArrayElement =317    Destruct318DestructArrayPart =319    DestructArrayElement320|   DestructRest321322DestructRest =323    '...'324    into:Name?325326DestructObjectField =327    field:Name328    (329        ':'330        Destruct331    )?332    (333        '='334        Expr335    )?336337// Aliases used to resolve node type conflicts338TrueExpr=Expr339FalseExpr=Expr340341// Trivia - tokens which will be implicitly skipped for parser342Trivia =343    'LIT_WHITESPACE!'344|   'LIT_MULTI_LINE_COMMENT!'345|   'ERROR_COMMENT_TOO_SHORT!'346|   'ERROR_COMMENT_UNTERMINATED!'347|   'LIT_SINGLE_LINE_HASH_COMMENT!'348|   'LIT_SINGLE_LINE_SLASH_COMMENT!'349350CustomError =351    'ERROR_MISSING_TOKEN!'352|   'ERROR_UNEXPECTED_TOKEN!'353|   'ERROR_CUSTOM!'
after · crates/jrsonnet-rowan-parser/jsonnet.ungram
1// This file describes structure of jsonnet source code2// It is also used to generate files in src/generated34// Token names ending with `!` are considered meta, and handled specifically56SourceFile = Expr78SuffixIndex =9    '?'?10    '.'11    index:Name12SuffixIndexExpr =13    ('?' '.')?14    '['15    index:Expr16    ']'17SuffixSlice =18    SliceDesc19SuffixApply =20    ArgsDesc21    'tailstrict'?22Suffix =23    SuffixIndex24|   SuffixIndexExpr25|   SuffixSlice26|   SuffixApply2728StmtLocal =29    'local'30    (Bind (',' Bind)* ','?)31    ';'32StmtAssert =33    Assertion34    ';'35Stmt =36    StmtLocal37|   StmtAssert3839ExprBinary =40    lhs:Expr41    BinaryOperator42    rhs:Expr43ExprUnary =44    UnaryOperator45    rhs:Expr46ExprObjExtend =47    lhs:Expr48    rhs:Expr49ExprParened =50    '('51    Expr52    ')'5354ExprLiteral =55    Literal56ExprString =57    Text58ExprNumber =59    Number60ExprArray =61    '['62    (Expr (',' Expr)* ','?)?63    ']'64ExprObject =65    ObjBody66ExprArrayComp =67    '['68    Expr69    ','?70    CompSpec*71    ']'7273ExprImport =74    ImportKind Text7576ImportKind =77    'importstr'78|   'importbin'79|   'import'8081ExprVar =82    name:Name8384ExprIfThenElse =85    'if'86    cond:Expr87    'then'88    then:TrueExpr89    ('else' else_:FalseExpr)?9091ExprFunction =92    'function'93    '('94    ParamsDesc95    ')'96    Expr97ExprError =98    'error'99    Expr100101Expr =102    Stmt*103    ExprBase104    Suffix*105106ExprBase =107    ExprBinary108|   ExprUnary109|   ExprObjExtend110|   ExprParened111|   ExprString112|   ExprNumber113|   ExprLiteral114|   ExprArray115|   ExprObject116|   ExprArrayComp117|   ExprImport118|   ExprVar119|   ExprIfThenElse120|   ExprFunction121|   ExprError122123BinaryOperator =124    '||' | '??' | '&&'125|   '|' | '^' | '&'126|   '==' | '!=' | '<' | '>' | '<=' | '>=' | 'in'127|   '<<' | '>>'128|   '+' | '-'129|   '*' | '/' | '%'130|   'META_OBJECT_APPLY!'131|   'ERROR_NO_OPERATOR!'132133UnaryOperator =134    '-' | '!' | '~'135136SliceDescEnd=Expr137SliceDescStep=Expr138SliceDesc =139    '['140    from:Expr?141    ':'142    (143        end:SliceDescEnd?144        (145            ':'146            step:SliceDescStep?147        )?148    )?149    ']'150151Name =152    'LIT_IDENT!'153154ArgsDesc =155    '('156    (Arg (',' Arg)* ','?)?157    ')'158Arg =159    (name:Name '=')? Expr160161ObjBodyComp =162    '{'163    (MemberComp (',' MemberComp)* ','?)?164    CompSpec*165    '}'166ObjBodyMemberList =167    '{'168    (Member (',' Member)* ','?)?169    '}'170ObjBody =171    ObjBodyComp172|   ObjBodyMemberList173174MemberBindStmt = ObjLocal175MemberAssertStmt = Assertion176MemberFieldNormal =177    FieldName178    '+'?179    Visibility180    Expr181MemberFieldMethod =182    FieldName183    ParamsDesc184    Visibility185    Expr186MemberComp =187    MemberBindStmt188|    MemberFieldNormal189|   MemberFieldMethod190Member =191    MemberBindStmt192|   MemberAssertStmt193|   MemberFieldNormal194|   MemberFieldMethod195196ObjLocal =197    'local'198    Bind199200FieldNameFixed =201    id:Name202|   Text203FieldNameDynamic =204    '['205    Expr206    ']'207FieldName =208    FieldNameFixed209|   FieldNameDynamic210211Visibility =212    ':' ':'? ':'?213214Literal =215    'null'216|   'true'217|   'false'218|   'self'219|   '$'220|   'super'221222Text =223    'LIT_STRING_DOUBLE!'224|   'ERROR_STRING_DOUBLE_UNTERMINATED!'225|   'LIT_STRING_SINGLE!'226|   'ERROR_STRING_SINGLE_UNTERMINATED!'227|   'LIT_STRING_DOUBLE_VERBATIM!'228|   'ERROR_STRING_DOUBLE_VERBATIM_UNTERMINATED!'229|   'LIT_STRING_SINGLE_VERBATIM!'230|   'ERROR_STRING_SINGLE_VERBATIM_UNTERMINATED!'231|   'ERROR_STRING_VERBATIM_MISSING_QUOTES!'232|   'LIT_STRING_BLOCK!'233|   'ERROR_STRING_BLOCK_UNEXPECTED_END!'234|   'ERROR_STRING_BLOCK_MISSING_NEW_LINE!'235|   'ERROR_STRING_BLOCK_MISSING_TERMINATION!'236|   'ERROR_STRING_BLOCK_MISSING_INDENT!'237238Number =239    'LIT_FLOAT!'240|   'ERROR_FLOAT_JUNK_AFTER_POINT!'241|   'ERROR_FLOAT_JUNK_AFTER_EXPONENT!'242|   'ERROR_FLOAT_JUNK_AFTER_EXPONENT_SIGN!'243244ForSpec =245    'for'246    bind:Destruct247    'in'248    Expr249ForObjSpec =250    'for'251    '['252    key:Name253    ']'254    Visibility255    value:Destruct256    'in'257    Expr258IfSpec =259    'if'260    Expr261CompSpec =262    ForSpec263|   ForObjSpec264|   IfSpec265266BindDestruct =267    into:Destruct268    '='269    value:Expr270BindFunction =271    name:Name272    params:ParamsDesc273    '='274    value:Expr275Bind =276    BindDestruct277|   BindFunction278279ParamsDesc =280    '('281    (Param (',' Param)* ','?)?282    ')'283Param =284    Destruct285    (286        '='287        Expr288    )?289290Assertion =291    'assert'292    condition:Expr293    (294        ':'295        message:Expr296    )?297298DestructFull =299    Name300DestructSkip =301    '?'302DestructArray =303    '['304    (305        DestructArrayPart306        (',' DestructArrayPart)*307        ','?308    )?309    ']'310DestructObject =311    '{'312    (313        DestructObjectField314        (',' DestructObjectField)*315        ','?316    )?317    DestructRest?318    ','?319    '}'320Destruct =321    DestructFull322|   DestructSkip323|   DestructArray324|   DestructObject325326DestructArrayElement =327    Destruct328DestructArrayPart =329    DestructArrayElement330|   DestructRest331332DestructRest =333    '...'334    into:Name?335336DestructObjectField =337    field:Name338    (339        ':'340        Destruct341    )?342    (343        '='344        Expr345    )?346347// Aliases used to resolve node type conflicts348TrueExpr=Expr349FalseExpr=Expr350351// Trivia - tokens which will be implicitly skipped for parser352Trivia =353    'LIT_WHITESPACE!'354|   'LIT_MULTI_LINE_COMMENT!'355|   'ERROR_COMMENT_TOO_SHORT!'356|   'ERROR_COMMENT_UNTERMINATED!'357|   'LIT_SINGLE_LINE_HASH_COMMENT!'358|   'LIT_SINGLE_LINE_SLASH_COMMENT!'359360CustomError =361    'ERROR_MISSING_TOKEN!'362|   'ERROR_UNEXPECTED_TOKEN!'363|   'ERROR_CUSTOM!'
modifiedcrates/jrsonnet-rowan-parser/src/generated/nodes.rsdiffbeforeafterboth
--- a/crates/jrsonnet-rowan-parser/src/generated/nodes.rs
+++ b/crates/jrsonnet-rowan-parser/src/generated/nodes.rs
@@ -650,6 +650,37 @@
 }
 
 #[derive(Debug, Clone, PartialEq, Eq, Hash)]
+pub struct ForObjSpec {
+	pub(crate) syntax: SyntaxNode,
+}
+impl ForObjSpec {
+	pub fn for_kw_token(&self) -> Option<SyntaxToken> {
+		support::token(&self.syntax, T![for])
+	}
+	pub fn l_brack_token(&self) -> Option<SyntaxToken> {
+		support::token(&self.syntax, T!['['])
+	}
+	pub fn key(&self) -> Option<Name> {
+		support::children(&self.syntax).next()
+	}
+	pub fn r_brack_token(&self) -> Option<SyntaxToken> {
+		support::token(&self.syntax, T![']'])
+	}
+	pub fn visibility(&self) -> Option<Visibility> {
+		support::children(&self.syntax).next()
+	}
+	pub fn value(&self) -> Option<Destruct> {
+		support::children(&self.syntax).next()
+	}
+	pub fn in_kw_token(&self) -> Option<SyntaxToken> {
+		support::token(&self.syntax, T![in])
+	}
+	pub fn expr(&self) -> Option<Expr> {
+		support::children(&self.syntax).next()
+	}
+}
+
+#[derive(Debug, Clone, PartialEq, Eq, Hash)]
 pub struct IfSpec {
 	pub(crate) syntax: SyntaxNode,
 }
@@ -845,6 +876,7 @@
 #[derive(Debug, Clone, PartialEq, Eq, Hash)]
 pub enum CompSpec {
 	ForSpec(ForSpec),
+	ForObjSpec(ForObjSpec),
 	IfSpec(IfSpec),
 }
 
@@ -1702,6 +1734,21 @@
 		&self.syntax
 	}
 }
+impl AstNode for ForObjSpec {
+	fn can_cast(kind: SyntaxKind) -> bool {
+		kind == FOR_OBJ_SPEC
+	}
+	fn cast(syntax: SyntaxNode) -> Option<Self> {
+		if Self::can_cast(syntax.kind()) {
+			Some(Self { syntax })
+		} else {
+			None
+		}
+	}
+	fn syntax(&self) -> &SyntaxNode {
+		&self.syntax
+	}
+}
 impl AstNode for IfSpec {
 	fn can_cast(kind: SyntaxKind) -> bool {
 		kind == IF_SPEC
@@ -2014,6 +2061,11 @@
 		CompSpec::ForSpec(node)
 	}
 }
+impl From<ForObjSpec> for CompSpec {
+	fn from(node: ForObjSpec) -> CompSpec {
+		CompSpec::ForObjSpec(node)
+	}
+}
 impl From<IfSpec> for CompSpec {
 	fn from(node: IfSpec) -> CompSpec {
 		CompSpec::IfSpec(node)
@@ -2022,13 +2074,14 @@
 impl AstNode for CompSpec {
 	fn can_cast(kind: SyntaxKind) -> bool {
 		match kind {
-			FOR_SPEC | IF_SPEC => true,
+			FOR_SPEC | FOR_OBJ_SPEC | IF_SPEC => true,
 			_ => false,
 		}
 	}
 	fn cast(syntax: SyntaxNode) -> Option<Self> {
 		let res = match syntax.kind() {
 			FOR_SPEC => CompSpec::ForSpec(ForSpec { syntax }),
+			FOR_OBJ_SPEC => CompSpec::ForObjSpec(ForObjSpec { syntax }),
 			IF_SPEC => CompSpec::IfSpec(IfSpec { syntax }),
 			_ => return None,
 		};
@@ -2037,6 +2090,7 @@
 	fn syntax(&self) -> &SyntaxNode {
 		match self {
 			CompSpec::ForSpec(it) => &it.syntax,
+			CompSpec::ForObjSpec(it) => &it.syntax,
 			CompSpec::IfSpec(it) => &it.syntax,
 		}
 	}
@@ -3016,6 +3070,11 @@
 		std::fmt::Display::fmt(self.syntax(), f)
 	}
 }
+impl std::fmt::Display for ForObjSpec {
+	fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
+		std::fmt::Display::fmt(self.syntax(), f)
+	}
+}
 impl std::fmt::Display for IfSpec {
 	fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
 		std::fmt::Display::fmt(self.syntax(), f)
modifiedcrates/jrsonnet-rowan-parser/src/generated/syntax_kinds.rsdiffbeforeafterboth
--- a/crates/jrsonnet-rowan-parser/src/generated/syntax_kinds.rs
+++ b/crates/jrsonnet-rowan-parser/src/generated/syntax_kinds.rs
@@ -145,6 +145,7 @@
 	FIELD_NAME_FIXED,
 	FIELD_NAME_DYNAMIC,
 	FOR_SPEC,
+	FOR_OBJ_SPEC,
 	IF_SPEC,
 	BIND_DESTRUCT,
 	BIND_FUNCTION,
modifiedcrates/jrsonnet-rowan-parser/src/parser.rsdiffbeforeafterboth
--- a/crates/jrsonnet-rowan-parser/src/parser.rs
+++ b/crates/jrsonnet-rowan-parser/src/parser.rs
@@ -364,6 +364,19 @@
 fn compspec(p: &mut Parser) -> CompletedMarker {
 	assert!(p.at_ts(COMPSPEC));
 	if p.at(T![for]) {
+		if p.nth_at(1, T!['[']) && p.nth_at(2, IDENT) && p.nth_at(3, T![']']) && p.nth_at(4, T![:])
+		{
+			let m = p.start();
+			p.bump_assert(T![for]);
+			p.bump_assert(T!['[']);
+			name(p);
+			p.expect(T![']']);
+			visibility(p);
+			destruct(p);
+			p.expect(T![in]);
+			expr(p);
+			return m.complete(p, FOR_OBJ_SPEC);
+		}
 		let m = p.start();
 		p.bump();
 		destruct(p);
addedcrates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_force_visible.snapdiffbeforeafterboth
--- /dev/null
+++ b/crates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_force_visible.snap
@@ -0,0 +1,51 @@
+---
+source: crates/jrsonnet-rowan-parser/src/tests.rs
+expression: "{ [k]: v for [k]::: v in obj }\n"
+---
+SOURCE_FILE@0..31
+  EXPR@0..30
+    EXPR_OBJECT@0..30
+      OBJ_BODY_COMP@0..30
+        L_BRACE@0..1 "{"
+        WHITESPACE@1..2 " "
+        MEMBER_FIELD_NORMAL@2..8
+          FIELD_NAME_DYNAMIC@2..5
+            L_BRACK@2..3 "["
+            EXPR@3..4
+              EXPR_VAR@3..4
+                NAME@3..4
+                  IDENT@3..4 "k"
+            R_BRACK@4..5 "]"
+          VISIBILITY@5..6
+            COLON@5..6 ":"
+          WHITESPACE@6..7 " "
+          EXPR@7..8
+            EXPR_VAR@7..8
+              NAME@7..8
+                IDENT@7..8 "v"
+        WHITESPACE@8..9 " "
+        FOR_OBJ_SPEC@9..28
+          FOR_KW@9..12 "for"
+          WHITESPACE@12..13 " "
+          L_BRACK@13..14 "["
+          NAME@14..15
+            IDENT@14..15 "k"
+          R_BRACK@15..16 "]"
+          VISIBILITY@16..19
+            COLON@16..17 ":"
+            COLON@17..18 ":"
+            COLON@18..19 ":"
+          WHITESPACE@19..20 " "
+          DESTRUCT_FULL@20..21
+            NAME@20..21
+              IDENT@20..21 "v"
+          WHITESPACE@21..22 " "
+          IN_KW@22..24 "in"
+          WHITESPACE@24..25 " "
+          EXPR@25..28
+            EXPR_VAR@25..28
+              NAME@25..28
+                IDENT@25..28 "obj"
+        WHITESPACE@28..29 " "
+        R_BRACE@29..30 "}"
+  WHITESPACE@30..31 "\n"
addedcrates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_hidden.snapdiffbeforeafterboth
--- /dev/null
+++ b/crates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_hidden.snap
@@ -0,0 +1,50 @@
+---
+source: crates/jrsonnet-rowan-parser/src/tests.rs
+expression: "{ [k]: v for [k]:: v in obj }\n"
+---
+SOURCE_FILE@0..30
+  EXPR@0..29
+    EXPR_OBJECT@0..29
+      OBJ_BODY_COMP@0..29
+        L_BRACE@0..1 "{"
+        WHITESPACE@1..2 " "
+        MEMBER_FIELD_NORMAL@2..8
+          FIELD_NAME_DYNAMIC@2..5
+            L_BRACK@2..3 "["
+            EXPR@3..4
+              EXPR_VAR@3..4
+                NAME@3..4
+                  IDENT@3..4 "k"
+            R_BRACK@4..5 "]"
+          VISIBILITY@5..6
+            COLON@5..6 ":"
+          WHITESPACE@6..7 " "
+          EXPR@7..8
+            EXPR_VAR@7..8
+              NAME@7..8
+                IDENT@7..8 "v"
+        WHITESPACE@8..9 " "
+        FOR_OBJ_SPEC@9..27
+          FOR_KW@9..12 "for"
+          WHITESPACE@12..13 " "
+          L_BRACK@13..14 "["
+          NAME@14..15
+            IDENT@14..15 "k"
+          R_BRACK@15..16 "]"
+          VISIBILITY@16..18
+            COLON@16..17 ":"
+            COLON@17..18 ":"
+          WHITESPACE@18..19 " "
+          DESTRUCT_FULL@19..20
+            NAME@19..20
+              IDENT@19..20 "v"
+          WHITESPACE@20..21 " "
+          IN_KW@21..23 "in"
+          WHITESPACE@23..24 " "
+          EXPR@24..27
+            EXPR_VAR@24..27
+              NAME@24..27
+                IDENT@24..27 "obj"
+        WHITESPACE@27..28 " "
+        R_BRACE@28..29 "}"
+  WHITESPACE@29..30 "\n"
addedcrates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_value_destruct.snapdiffbeforeafterboth
--- /dev/null
+++ b/crates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_value_destruct.snap
@@ -0,0 +1,66 @@
+---
+source: crates/jrsonnet-rowan-parser/src/tests.rs
+expression: "{ [k]: a + b for [k]: [a, b] in obj }\n"
+---
+SOURCE_FILE@0..38
+  EXPR@0..37
+    EXPR_OBJECT@0..37
+      OBJ_BODY_COMP@0..37
+        L_BRACE@0..1 "{"
+        WHITESPACE@1..2 " "
+        MEMBER_FIELD_NORMAL@2..12
+          FIELD_NAME_DYNAMIC@2..5
+            L_BRACK@2..3 "["
+            EXPR@3..4
+              EXPR_VAR@3..4
+                NAME@3..4
+                  IDENT@3..4 "k"
+            R_BRACK@4..5 "]"
+          VISIBILITY@5..6
+            COLON@5..6 ":"
+          WHITESPACE@6..7 " "
+          EXPR@7..12
+            EXPR_BINARY@7..12
+              EXPR@7..8
+                EXPR_VAR@7..8
+                  NAME@7..8
+                    IDENT@7..8 "a"
+              WHITESPACE@8..9 " "
+              PLUS@9..10 "+"
+              WHITESPACE@10..11 " "
+              EXPR@11..12
+                EXPR_VAR@11..12
+                  NAME@11..12
+                    IDENT@11..12 "b"
+        WHITESPACE@12..13 " "
+        FOR_OBJ_SPEC@13..35
+          FOR_KW@13..16 "for"
+          WHITESPACE@16..17 " "
+          L_BRACK@17..18 "["
+          NAME@18..19
+            IDENT@18..19 "k"
+          R_BRACK@19..20 "]"
+          VISIBILITY@20..21
+            COLON@20..21 ":"
+          WHITESPACE@21..22 " "
+          DESTRUCT_ARRAY@22..28
+            L_BRACK@22..23 "["
+            DESTRUCT_FULL@23..24
+              NAME@23..24
+                IDENT@23..24 "a"
+            COMMA@24..25 ","
+            WHITESPACE@25..26 " "
+            DESTRUCT_FULL@26..27
+              NAME@26..27
+                IDENT@26..27 "b"
+            R_BRACK@27..28 "]"
+          WHITESPACE@28..29 " "
+          IN_KW@29..31 "in"
+          WHITESPACE@31..32 " "
+          EXPR@32..35
+            EXPR_VAR@32..35
+              NAME@32..35
+                IDENT@32..35 "obj"
+        WHITESPACE@35..36 " "
+        R_BRACE@36..37 "}"
+  WHITESPACE@37..38 "\n"
addedcrates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_visible.snapdiffbeforeafterboth
--- /dev/null
+++ b/crates/jrsonnet-rowan-parser/src/snapshots/jrsonnet_rowan_parser__tests__for_obj_spec_visible.snap
@@ -0,0 +1,49 @@
+---
+source: crates/jrsonnet-rowan-parser/src/tests.rs
+expression: "{ [k]: v for [k]: v in obj }\n"
+---
+SOURCE_FILE@0..29
+  EXPR@0..28
+    EXPR_OBJECT@0..28
+      OBJ_BODY_COMP@0..28
+        L_BRACE@0..1 "{"
+        WHITESPACE@1..2 " "
+        MEMBER_FIELD_NORMAL@2..8
+          FIELD_NAME_DYNAMIC@2..5
+            L_BRACK@2..3 "["
+            EXPR@3..4
+              EXPR_VAR@3..4
+                NAME@3..4
+                  IDENT@3..4 "k"
+            R_BRACK@4..5 "]"
+          VISIBILITY@5..6
+            COLON@5..6 ":"
+          WHITESPACE@6..7 " "
+          EXPR@7..8
+            EXPR_VAR@7..8
+              NAME@7..8
+                IDENT@7..8 "v"
+        WHITESPACE@8..9 " "
+        FOR_OBJ_SPEC@9..26
+          FOR_KW@9..12 "for"
+          WHITESPACE@12..13 " "
+          L_BRACK@13..14 "["
+          NAME@14..15
+            IDENT@14..15 "k"
+          R_BRACK@15..16 "]"
+          VISIBILITY@16..17
+            COLON@16..17 ":"
+          WHITESPACE@17..18 " "
+          DESTRUCT_FULL@18..19
+            NAME@18..19
+              IDENT@18..19 "v"
+          WHITESPACE@19..20 " "
+          IN_KW@20..22 "in"
+          WHITESPACE@22..23 " "
+          EXPR@23..26
+            EXPR_VAR@23..26
+              NAME@23..26
+                IDENT@23..26 "obj"
+        WHITESPACE@26..27 " "
+        R_BRACE@27..28 "}"
+  WHITESPACE@28..29 "\n"
modifiedcrates/jrsonnet-rowan-parser/src/tests.rsdiffbeforeafterboth
--- a/crates/jrsonnet-rowan-parser/src/tests.rs
+++ b/crates/jrsonnet-rowan-parser/src/tests.rs
@@ -228,6 +228,19 @@
 	local_in_binop_rhs => r#"
 		a + local x = 1; x
 	"#
+
+	for_obj_spec_visible => r#"
+		{ [k]: v for [k]: v in obj }
+	"#
+	for_obj_spec_hidden => r#"
+		{ [k]: v for [k]:: v in obj }
+	"#
+	for_obj_spec_force_visible => r#"
+		{ [k]: v for [k]::: v in obj }
+	"#
+	for_obj_spec_value_destruct => r#"
+		{ [k]: a + b for [k]: [a, b] in obj }
+	"#
 );
 
 #[test]