Skip to content

Commit

Permalink
docs: initial storage cleanup (#2433)
Browse files Browse the repository at this point in the history
Combine storage and state variable docs and extends with missing functions.
  • Loading branch information
LHerskind authored Sep 25, 2023
1 parent 922fc99 commit d833483
Show file tree
Hide file tree
Showing 19 changed files with 625 additions and 399 deletions.
2 changes: 1 addition & 1 deletion docs/docs/concepts/foundation/state_model.md
Original file line number Diff line number Diff line change
Expand Up @@ -40,4 +40,4 @@ This is achieved with two main features:

## Further reading

Read more about how to leverage the Aztec state model in Aztec contracts [here](../../dev_docs/contracts/syntax/state_variables.md).
Read more about how to leverage the Aztec state model in Aztec contracts [here](../../dev_docs/contracts/syntax/storage.md).
2 changes: 1 addition & 1 deletion docs/docs/dev_docs/contracts/layout.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
title: Structure
---

A contract is a collection of persistent [state variables](./syntax/state_variables.md), and [functions](./syntax/functions) which may manipulate these variables. Functions and state variables within a contract's scope are said to belong to that contract. A contract can only access and modify its own state. If a contract wishes to access or modify another contract's state, it must make a call to an external function of the other contract. For anything to happen on the Aztec network, an external function of a contract needs to be called.
A contract is a collection of persistent [state variables](./syntax/storage.md), and [functions](./syntax/functions) which may manipulate these variables. Functions and state variables within a contract's scope are said to belong to that contract. A contract can only access and modify its own state. If a contract wishes to access or modify another contract's state, it must make a call to an external function of the other contract. For anything to happen on the Aztec network, an external function of a contract needs to be called.

# Contract

Expand Down
2 changes: 1 addition & 1 deletion docs/docs/dev_docs/contracts/syntax/main.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ On top of [Noir's stdlib](https://noir-lang.org/standard_library/array_methods),

Aztec.nr contains abstractions which remove the need to understand the low-level Aztec protocol. Notably, it provides:

- Public and private [state variable types](./state_variables.md)
- Public and private [state variable types](./storage.md)
- Some pre-designed notes
- Functions for [emitting](./events.md) encrypted and unencrypted logs
- [Oracle functions](./functions.md#oracle-functions) for accessing:
Expand Down
370 changes: 0 additions & 370 deletions docs/docs/dev_docs/contracts/syntax/state_variables.md

This file was deleted.

528 changes: 510 additions & 18 deletions docs/docs/dev_docs/contracts/syntax/storage.md

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion docs/docs/dev_docs/tutorials/writing_token_contract.md
Original file line number Diff line number Diff line change
Expand Up @@ -242,7 +242,7 @@ We are also importing types from a `types.nr` file. Copy [this file](https://git

### Note on private state

Private state in Aztec is all [UTXOs](https://en.wikipedia.org/wiki/Unspent_transaction_output) under the hood. Handling UTXOs is largely abstracted away from developers, but there are some unique things for developers to be aware of when creating and managing private state in an Aztec contract. See [State Variables](../contracts/syntax/state_variables) to learn more about public and private state in Aztec.
Private state in Aztec is all [UTXOs](https://en.wikipedia.org/wiki/Unspent_transaction_output) under the hood. Handling UTXOs is largely abstracted away from developers, but there are some unique things for developers to be aware of when creating and managing private state in an Aztec contract. See [State Variables](../contracts/syntax/storage.md) to learn more about public and private state in Aztec.

Copy [`util.nr`](https://github.com/AztecProtocol/aztec-packages/blob/master/yarn-project/noir-contracts/src/contracts/token_contract/src/util.nr) into `token_contract_tutorial/contracts/src` as well. The function defined in `util.nr` will be helpful for generating message hashes that are used when communicating between contracts.

Expand Down
8 changes: 6 additions & 2 deletions docs/netlify.toml
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@

[[redirects]]
from = "/dev_docs/contracts/state_variables"
to = "/dev_docs/contracts/syntax/state_variables"
to = "/dev_docs/contracts/syntax/storage"

[[redirects]]
from = "/dev_docs/contracts/functions"
Expand Down Expand Up @@ -164,4 +164,8 @@

[[redirects]]
from = "/aztec/*"
to = "/"
to = "/"

[[redirects]]
from = "/dev_docs/contracts/syntax/state_variables"
to = "/dev_docs/contracts/syntax/storage"
1 change: 0 additions & 1 deletion docs/sidebars.js
Original file line number Diff line number Diff line change
Expand Up @@ -248,7 +248,6 @@ const sidebars = {
},
items: [
"dev_docs/contracts/syntax/storage",
"dev_docs/contracts/syntax/state_variables",
"dev_docs/contracts/syntax/events",
"dev_docs/contracts/syntax/functions",
"dev_docs/contracts/syntax/context",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,16 @@ use crate::oracle;
use dep::std::hash::pedersen_with_separator;
use dep::std::option::Option;

// docs:start:struct
struct ImmutableSingleton<Note, N> {
context: Context,
storage_slot: Field,
note_interface: NoteInterface<Note, N>,
}
// docs:end:struct

impl<Note, N> ImmutableSingleton<Note, N> {
// docs:start:new
fn new(
context: Context,
storage_slot: Field,
Expand All @@ -29,12 +32,14 @@ impl<Note, N> ImmutableSingleton<Note, N> {
note_interface,
}
}
// docs:end:new

unconstrained fn is_initialized(self) -> bool {
let nullifier = self.compute_initialisation_nullifier();
oracle::notes::is_nullifier_emitted(nullifier)
}

// docs:start:initialize
fn initialize(self, note: &mut Note) {
// Nullify the storage slot.
let nullifier = self.compute_initialisation_nullifier();
Expand All @@ -49,6 +54,7 @@ impl<Note, N> ImmutableSingleton<Note, N> {
self.note_interface,
);
}
// docs:end:initialize

fn compute_initialisation_nullifier(self) -> Field {
pedersen_with_separator(
Expand All @@ -57,10 +63,12 @@ impl<Note, N> ImmutableSingleton<Note, N> {
)[0]
}

// docs:start:get_note
fn get_note(self) -> Note {
let storage_slot = self.storage_slot;
get_note(self.context.private.unwrap(), storage_slot, self.note_interface)
}
// docs:end:get_note

unconstrained fn view_note(self) -> Note {
let options = NoteViewerOptions::new().set_limit(1);
Expand Down
6 changes: 6 additions & 0 deletions yarn-project/aztec-nr/aztec/src/state_vars/map.nr
Original file line number Diff line number Diff line change
@@ -1,13 +1,16 @@
use crate::context::{PrivateContext, PublicContext, Context};
use dep::std::option::Option;

// docs:start:map
struct Map<V> {
context: Context,
storage_slot: Field,
state_var_constructor: fn(Context, Field) -> V,
}
// docs:end:map

impl<V> Map<V> {
// docs:start:new
fn new(
context: Context,
storage_slot: Field,
Expand All @@ -20,12 +23,15 @@ impl<V> Map<V> {
state_var_constructor,
}
}
// docs:end:new

// docs:start:at
fn at(self, key: Field) -> V {
// TODO(#1204): use a generator index for the storage slot
let derived_storage_slot = dep::std::hash::pedersen([self.storage_slot, key])[0];

let state_var_constructor = self.state_var_constructor;
state_var_constructor(self.context, derived_storage_slot)
}
// docs:end:at
}
8 changes: 8 additions & 0 deletions yarn-project/aztec-nr/aztec/src/state_vars/public_state.nr
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,15 @@ use crate::oracle::storage::storage_write;
use crate::types::type_serialization::TypeSerializationInterface;
use dep::std::option::Option;

// docs:start:public_state_struct
struct PublicState<T, T_SERIALIZED_LEN> {
storage_slot: Field,
serialization_methods: TypeSerializationInterface<T, T_SERIALIZED_LEN>,
}
// docs:end:public_state_struct

impl<T, T_SERIALIZED_LEN> PublicState<T, T_SERIALIZED_LEN> {
// docs:start:public_state_struct_new
fn new(
// Note: Passing the contexts to new(...) just to have an interface compatible with a Map.
_: Context,
Expand All @@ -22,14 +25,19 @@ impl<T, T_SERIALIZED_LEN> PublicState<T, T_SERIALIZED_LEN> {
serialization_methods,
}
}
// docs:end:public_state_struct_new

// docs:start:public_state_struct_read
fn read(self) -> T {
storage_read(self.storage_slot, self.serialization_methods.deserialize)
}
// docs:end:public_state_struct_read

// docs:start:public_state_struct_write
fn write(self, value: T) {
let serialize = self.serialization_methods.serialize;
let fields = serialize(value);
storage_write(self.storage_slot, fields);
}
// docs:end:public_state_struct_write
}
20 changes: 19 additions & 1 deletion yarn-project/aztec-nr/aztec/src/state_vars/set.nr
Original file line number Diff line number Diff line change
Expand Up @@ -12,13 +12,16 @@ use crate::note::{
utils::compute_note_hash_for_read_or_nullify,
};

// docs:start:struct
struct Set<Note, N> {
context: Context,
storage_slot: Field,
note_interface: NoteInterface<Note, N>,
}
// docs:end:struct

impl<Note, N> Set<Note, N> {
// docs:start:new
fn new(
context: Context,
storage_slot: Field,
Expand All @@ -31,7 +34,9 @@ impl<Note, N> Set<Note, N> {
note_interface,
}
}
// docs:end:new

// docs:start:insert
fn insert(self, note: &mut Note) {
create_note(
self.context.private.unwrap(),
Expand All @@ -40,7 +45,9 @@ impl<Note, N> Set<Note, N> {
self.note_interface,
);
}
// docs:end:insert

// docs:start:insert_from_public
fn insert_from_public(self, note: &mut Note) {
create_note_hash_from_public(
self.context.public.unwrap(),
Expand All @@ -49,7 +56,9 @@ impl<Note, N> Set<Note, N> {
self.note_interface,
);
}

// docs:end:insert_from_public

// docs:start:assert_contains_and_remove
fn assert_contains_and_remove(self, note: &mut Note, nonce: Field) {
// Initialize header of note. Must be done before computing note hashes as it initializes the:
// - storage slot (used in inner note hash)
Expand Down Expand Up @@ -77,7 +86,9 @@ impl<Note, N> Set<Note, N> {
self.note_interface,
);
}
// docs:end:assert_contains_and_remove

// docs:start:assert_contains_and_remove_publicly_created
// NOTE: this function should ONLY be used for PUBLICLY-CREATED note hashes!
// WARNING: function will be deprecated/removed eventually once public kernel applies nonces.
fn assert_contains_and_remove_publicly_created(self, note: &mut Note) {
Expand Down Expand Up @@ -114,7 +125,9 @@ impl<Note, N> Set<Note, N> {
self.note_interface,
);
}
// docs:end:assert_contains_and_remove_publicly_created

// docs:start:remove
fn remove(self, note: Note) {
let context = self.context.private.unwrap();
let note_hash = compute_note_hash_for_read_or_nullify(self.note_interface, note);
Expand All @@ -127,7 +140,9 @@ impl<Note, N> Set<Note, N> {
self.note_interface,
);
}
// docs:end:remove

// docs:start:get_notes
fn get_notes<FILTER_ARGS>(
self,
options: NoteGetterOptions<Note, N, FILTER_ARGS>,
Expand All @@ -141,11 +156,14 @@ impl<Note, N> Set<Note, N> {
);
opt_notes
}
// docs:end:get_notes

// docs:start:view_notes
unconstrained fn view_notes(
self,
options: NoteViewerOptions<Note, N>,
) -> [Option<Note>; MAX_NOTES_PER_PAGE] {
view_notes(self.storage_slot, self.note_interface, options)
}
// docs:end:view_notes
}
10 changes: 10 additions & 0 deletions yarn-project/aztec-nr/aztec/src/state_vars/singleton.nr
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,16 @@ use crate::oracle;
use dep::std::hash::pedersen_with_separator;
use dep::std::option::Option;

// docs:start:struct
struct Singleton<Note, N> {
context: Option<&mut PrivateContext>,
storage_slot: Field,
note_interface: NoteInterface<Note, N>,
}
// docs:end:struct

impl<Note, N> Singleton<Note, N> {
// docs:start:new
fn new(
context: Context,
storage_slot: Field,
Expand All @@ -29,12 +32,14 @@ impl<Note, N> Singleton<Note, N> {
note_interface,
}
}
// docs:end:new

unconstrained fn is_initialized(self) -> bool {
let nullifier = self.compute_initialisation_nullifier();
oracle::notes::is_nullifier_emitted(nullifier)
}

// docs:start:initialize
fn initialize(self, note: &mut Note) {
let context = self.context.unwrap();
// Nullify the storage slot.
Expand All @@ -43,6 +48,7 @@ impl<Note, N> Singleton<Note, N> {

create_note(context, self.storage_slot, note, self.note_interface);
}
// docs:end:initialize

fn compute_initialisation_nullifier(self) -> Field {
pedersen_with_separator(
Expand All @@ -51,6 +57,7 @@ impl<Note, N> Singleton<Note, N> {
)[0]
}

// docs:start:replace
fn replace(self, new_note: &mut Note) {
let context = self.context.unwrap();
let prev_note = get_note(context, self.storage_slot, self.note_interface);
Expand All @@ -61,7 +68,9 @@ impl<Note, N> Singleton<Note, N> {
// Add replacement note.
create_note(context, self.storage_slot, new_note, self.note_interface);
}
// docs:end:replace

// docs:start:get_note
fn get_note(self) -> Note {
let context = self.context.unwrap();
let mut note = get_note(context, self.storage_slot, self.note_interface);
Expand All @@ -75,6 +84,7 @@ impl<Note, N> Singleton<Note, N> {

note
}
// docs:end:get_note

unconstrained fn view_note(self) -> Note {
let options = NoteViewerOptions::new().set_limit(1);
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use crate::types::type_serialization::TypeSerializationInterface;

// docs:start:field_serialization
global FIELD_SERIALIZED_LEN: Field = 1;

fn deserializeField(fields: [Field; FIELD_SERIALIZED_LEN]) -> Field {
Expand All @@ -13,4 +14,5 @@ fn serializeField(value: Field) -> [Field; FIELD_SERIALIZED_LEN] {
global FieldSerializationMethods = TypeSerializationInterface {
deserialize: deserializeField,
serialize: serializeField,
};
};
// docs:end:field_serialization
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,9 @@ impl EasyPrivateUint {
let mut addend_note = ValueNote::new(addend as Field, owner);

// Insert the new note to the owner's set of notes.
// docs:start:insert
self.set.insert(&mut addend_note);
// docs:end:insert

// Emit the newly created encrypted note preimages via oracle calls.
let owner_key = get_public_key(owner);
Expand All @@ -58,8 +60,10 @@ impl EasyPrivateUint {

// Very similar to `value_note::utils::decrement`.
fn sub(self, subtrahend: u120, owner: Field) {
// docs:start:get_notes
let options = NoteGetterOptions::with_filter(filter_notes_min_sum, subtrahend as Field);
let maybe_notes = self.set.get_notes(options);
// docs:end:get_notes

let mut minuend: u120 = 0;
for i in 0..maybe_notes.len() {
Expand All @@ -71,7 +75,9 @@ impl EasyPrivateUint {
assert(note.owner == owner);

// Removes the note from the owner's set of notes.
// docs:start:remove
self.set.remove(note);
// docs:end:remove

minuend += note.value as u120;
}
Expand Down
Loading

0 comments on commit d833483

Please sign in to comment.