Skip to content
This repository was archived by the owner on Mar 17, 2025. It is now read-only.

Commit 3e263f6

Browse files
committed
Merge pull request #112 from firebase/koss-function-deprecation-merge
Deprecate function format fn(x) = exp.
2 parents 18d357f + 8d7b940 commit 3e263f6

18 files changed

+155
-145
lines changed

docs/changelog.txt

+1
Original file line numberDiff line numberDiff line change
@@ -1 +1,2 @@
11
changed - Path expressions should now use path template syntax instead of wildcard variables.
2+
changed - Deprecate fn(x) = exp; function definition format (with deprecation warning).

docs/guide.md

+39-39
Original file line numberDiff line numberDiff line change
@@ -29,12 +29,12 @@ permissions can be written as:
2929
[all_access.bolt](../samples/all_access.bolt)
3030
```javascript
3131
path / {
32-
read() = true;
33-
write() = true;
32+
read() { true }
33+
write() { true }
3434
}
3535
```
3636

37-
The `read() = true` and `write() = true` methods allow everyone to read and write this location
37+
The `read() { true }` and `write() { true }` methods allow everyone to read and write this location
3838
(and all children under this location). You can also use more complex expressions instead of
3939
`true`. When the expression evaluates to `true` the read or write operation is allowed.
4040

@@ -97,16 +97,16 @@ _posts.bolt_
9797
```javascript
9898
// Allow anyone to read the list of Posts.
9999
path /posts {
100-
read() = true;
100+
read() { true }
101101
}
102102

103103
// All individual Posts are writable by anyone.
104104
path /posts/{id} is Post {
105-
write() = true;
105+
write() { true }
106106
}
107107

108108
type Post {
109-
validate() = this.message.length <= 140;
109+
validate() { this.message.length <= 140 }
110110

111111
message: String,
112112
from: String
@@ -124,7 +124,7 @@ captured variable `id` being equal to (the string) '123'.
124124
The Post type allows for exactly two string properties in each post (message and
125125
from). It also ensures that no message is longer than 140 characters.
126126

127-
Bolt type statements can contain a `validate()` method (defined as `validate() = <expression>`,
127+
Bolt type statements can contain a `validate()` method (defined as `validate() { <expression> }`,
128128
where the expression evaluates to `true` if the data is valid (can be saved to the
129129
database). When the expression evaluates to `false`, the attempt to write the data will return
130130
an error to the Firebase client and the database will be unmodified.
@@ -220,7 +220,7 @@ type Room {
220220
}
221221

222222
type NameString extends String {
223-
validate() = this.length > 0 && this.length <= 32;
223+
validate() { this.length > 0 && this.length <= 32 }
224224
}
225225
```
226226

@@ -273,8 +273,8 @@ definitions look just like _type_ and _path_ methods, except they can also accep
273273

274274
```javascript
275275
path /users/{userid} is User {
276-
read() = true;
277-
write() = isCurrentUser(userid);
276+
read() { true }
277+
write() { isCurrentUser(userid) }
278278
}
279279

280280
type User {
@@ -284,7 +284,7 @@ type User {
284284

285285
// Define isCurrentUser() function to test if the given user id
286286
// matches the currently signed-in user.
287-
isCurrentUser(uid) = auth != null && auth.uid == uid;
287+
isCurrentUser(uid) { auth != null && auth.uid == uid }
288288
```
289289

290290
```JSON
@@ -326,7 +326,7 @@ path /posts/{id} is Post;
326326

327327
type Post {
328328
// Make sure that the only value allowed to be written is now.
329-
validate() = this.modified == now;
329+
validate() { this.modified == now }
330330

331331
message: String,
332332
modified: Number
@@ -340,8 +340,8 @@ A handy way to express this is to use a user-defined type for the CurrentTimesta
340340

341341
```javascript
342342
path /posts/{id} is Post {
343-
read() = true;
344-
write() = true;
343+
read() { true }
344+
write() { true }
345345
}
346346

347347
type Post {
@@ -350,7 +350,7 @@ type Post {
350350
}
351351

352352
type CurrentTimestamp extends Number {
353-
validate() = this == now;
353+
validate() { this == now }
354354
}
355355
```
356356

@@ -359,8 +359,8 @@ when first written, and never change thereafter:
359359

360360
```javascript
361361
path /posts/{id} is Post {
362-
read() = true;
363-
write() = true;
362+
read() { true }
363+
write() { true }
364364
}
365365

366366
type Post {
@@ -370,16 +370,16 @@ type Post {
370370
}
371371

372372
type CurrentTimestamp extends Number {
373-
validate() = this == now;
373+
validate() { this == now }
374374
}
375375

376376
type InitialTimestamp extends Number {
377-
validate() = initial(this, now);
377+
validate() { initial(this, now) }
378378
}
379379

380380
// Returns true if the value is intialized to init, or if it retains it's prior
381381
// value, otherwise.
382-
initial(value, init) = value == (prior(value) == null ? init : prior(value));
382+
initial(value, init) { value == (prior(value) == null ? init : prior(value)) }
383383
```
384384

385385
Note the special function `prior(ref)` - returns the previous value stored at a given database location
@@ -419,8 +419,8 @@ to define the Timestamp example above is:
419419
```javascript
420420
// Note the use of Timestamped version of a Post type.
421421
path /posts/{id} is Timestamped<Post> {
422-
read() = true;
423-
write() = true;
422+
read() { true }
423+
write() { true }
424424
}
425425

426426
type Post {
@@ -433,16 +433,16 @@ type Timestamped<T> extends T {
433433
}
434434

435435
type CurrentTimestamp extends Number {
436-
validate() = this == now;
436+
validate() { this == now }
437437
}
438438

439439
type InitialTimestamp extends Number {
440-
validate() = initial(this, now);
440+
validate() { initial(this, now) }
441441
}
442442

443443
// Returns true if the value is intialized to init, or retains it's prior
444444
// value, otherwise.
445-
initial(value, init) = value == (prior(value) == null ? init : prior(value));
445+
initial(value, init) { value == (prior(value) == null ? init : prior(value)) }
446446
```
447447

448448
```JSON
@@ -481,34 +481,34 @@ Rules](https://www.firebase.com/docs/security/guide/user-security.html#section-r
481481
// Room Names
482482
//
483483
path /rooms_names is String[] {
484-
read() = isSignedIn();
484+
read() { isSignedIn() }
485485
}
486486

487-
getRoomName(id) = prior(root.room_names[id]);
487+
getRoomName(id) { prior(root.room_names[id]) }
488488

489489
//
490490
// Room Members
491491
//
492492
path /members/{room_id} {
493-
read() = isRoomMember(room_id);
493+
read() { isRoomMember(room_id) }
494494
}
495495

496496
path /members/{room_id}/{user_id} is NameString {
497-
write() = isCurrentUser(user_id);
497+
write() { isCurrentUser(user_id) }
498498
}
499499

500-
isRoomMember(room_id) = isSignedIn() && prior(root.members[room_id][auth.uid]) != null;
500+
isRoomMember(room_id) { isSignedIn() && prior(root.members[room_id][auth.uid]) != null }
501501

502502
//
503503
// Messages
504504
//
505505
path /messages/{room_id} {
506-
read() = isRoomMember(room_id);
507-
validate() = getRoomName(room_id) != null;
506+
read() { isRoomMember(room_id) }
507+
validate() { getRoomName(room_id) != null }
508508
}
509509

510510
path /messages/{room_id}/{message_id} is Message {
511-
write() = createOnly(this) && isRoomMember(room_id);
511+
write() { createOnly(this) && isRoomMember(room_id) }
512512
}
513513

514514
type Message {
@@ -518,26 +518,26 @@ type Message {
518518
}
519519

520520
type MessageString extends String {
521-
validate() = this.length > 0 && this.length < 50;
521+
validate() { this.length > 0 && this.length < 50 }
522522
}
523523

524524
//
525525
// Helper Types
526526
//
527527
type CurrentTimestamp extends Number {
528-
validate() = this == now;
528+
validate() { this == now }
529529
}
530530

531531
type NameString {
532-
validate() = this.length > 0 && this.length < 20;
532+
validate() { this.length > 0 && this.length < 20 }
533533
}
534534

535535
//
536536
// Helper Functions
537537
//
538-
isCurrentUser(uid) = isSignedIn() && auth.uid == uid;
539-
isSignedIn() = auth != null;
540-
createOnly(value) = prior(value) == null && value != null;
538+
isCurrentUser(uid) { isSignedIn() && auth.uid == uid }
539+
isSignedIn() { auth != null }
540+
createOnly(value) { prior(value) == null && value != null }
541541
```
542542

543543
```JSON

docs/language.md

+16-16
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ type MyType [extends BaseType] {
3333
property2: Type,
3434
...
3535

36-
validate() = <validation expression>;
36+
validate() { <validation expression> }
3737
}
3838
}
3939
```
@@ -67,11 +67,11 @@ Any of the built-in scalar types can be _extended_ by adding a validation expres
6767

6868
```javascript
6969
type ShortString extends String {
70-
validate() = this.length < 32;
70+
validate() { this.length < 32 }
7171
}
7272

7373
type Percentage extends Number {
74-
validate() = this >=0 && this <= 100;
74+
validate() { this >=0 && this <= 100 }
7575
}
7676
```
7777

@@ -103,7 +103,7 @@ type Model {
103103
}
104104

105105
type ProductID extends String {
106-
validate() = this.length <= 20;
106+
validate() { this.length <= 20 }
107107
}
108108
```
109109

@@ -145,11 +145,11 @@ A path statement provides access and validation rules for data stored at a given
145145

146146
```javascript
147147
path /path/to/data [is Type] {
148-
read() = <true-iff-reading-this-path-is-allowed>;
148+
read() { <true-iff-reading-this-path-is-allowed> }
149149

150-
write() = <true-iff-writing-this-path-is-allowed>;
150+
write() { <true-iff-writing-this-path-is-allowed> }
151151

152-
validate() = <additional-validation-rules>;
152+
validate() { <additional-validation-rules> }
153153
}
154154
```
155155

@@ -174,10 +174,10 @@ an expression as a variable parameter:
174174
```javascript
175175
path /users/{uid} is User {
176176
// Anyone can read a User's information.
177-
read() = true;
177+
read() { true }
178178

179179
// Only an authenticated user can write their information.
180-
write() = auth != null && auth.uid == uid;
180+
write() { auth != null && auth.uid == uid }
181181
}
182182
```
183183

@@ -237,7 +237,7 @@ and one of its properties:
237237
path /products is Product[];
238238

239239
type Product {
240-
validate() = this.id == key();
240+
validate() { this.id == key() }
241241

242242
id: String,
243243
name: String
@@ -258,16 +258,16 @@ function isUser(uid) {
258258

259259
function isUser(uid) { auth != null && auth.uid == uid }
260260

261-
isUser(uid) = auth != null && auth.uid == uid;
261+
isUser(uid) { auth != null && auth.uid == uid }
262262
```
263263

264264
Similarly, methods in path and type statements can use the abbreviated functional form (all
265265
these are equivalent):
266266

267267
```javascript
268268
write() { return this.user == auth.uid; }
269+
write() { this.user == auth.uid; }
269270
write() { this.user == auth.uid }
270-
write() = this.user == auth.uid;
271271
```
272272

273273
# Expressions
@@ -295,10 +295,10 @@ not identical in Bolt. This section demonstrates how equivalent behavior is ach
295295

296296
API | Bolt Equivalent
297297
----| ---------------
298-
".read" : "exp" | read() { return exp; }
299-
".write" : "exp" | write() { return exp; }
300-
".validate": "exp" | validate() { return exp; }
301-
".indexOn": [ "prop", ...] | index() { return [ "prop", ... ] }
298+
".read" : "exp" | read() { exp }
299+
".write" : "exp" | write() { exp }
300+
".validate": "exp" | validate() { exp }
301+
".indexOn": [ "prop", ...] | index() { [ "prop", ... ] }
302302

303303
## Variables
304304

samples/all_access.bolt

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
// Equivalent to the default (most-permissive) Firebase rule.
22
path / {
3-
read() = true;
4-
write() = true;
3+
read() { true }
4+
write() { true }
55
}

0 commit comments

Comments
 (0)