Skip to content

Latest commit

 

History

History
535 lines (427 loc) · 15.1 KB

dumping.livemd

File metadata and controls

535 lines (427 loc) · 15.1 KB

Dumping

Mix.install([
  {:kino_vega_lite, "~> 0.1.10"}
])

Index

  1. Toc
  2. Contributing
    1. Understanding Any Module
    2. Style Guide
    3. Writing Documents
    4. Examples Over Testing
    5. Git
    6. Iex
    7. Mnesia Vs Actor State
    8. Observer
    9. Testing
      1. Running Tests
      2. Writing Tests
  3. Visualization
    1. Actors
  4. Hoon
    1. Calling
    2. Dumping
    3. Setting Up
  5. Analysis
  6. Jam

Dumping Nock

Given a functioning Hoon environment with Anoma loaded, we can now start dumping various data in the environment.

This guide hopefully serves as a good way to give you the tools needed to dump anything for yourself.

Dumping Functions

Dumping any Hoon gate is relatively easy.

However, first we need to learn how to get Hoon to let us use Nock properly. A good way is by reading the dot(.) section, as runes starting with . deal with nock operations.

In particular we wish to focus on dottar(.*), which deals with calling nock on some expression.

We won't go into detail about calling functions in this section, however there is another section that focuses solely on how to call functions and how it works in Nock.

Speaking of functions, we should know just a few things about the layout of functions, and their important indicies.

Functions in Hoon are laid out as the following:

[function sample environment-defined-in]
  • Function is some nock logic we wish to run
  • Sample is the default argument of the function if non is given
  • Environment-defined-in is the environment the function is defined in and relies upon.

A good basic example can be seen below:

[[0 6] 777 999]

This function has an arbitrary environment of 999 and a sample of 777. The logic itself simply grabs the sample from the environment.

A good visualization of the indexing can be seen below

stateDiagram-v2
1 --> 2
1 --> 3
2 --> 4
2 --> 5
3 --> 6
3 --> 7
Loading

where we have in our concrete example

Index Nock
1 [[0 6] 777 999]
2 [0 6]
3 [777 999]
4 0
5 6
6 777
7 999

With some basics out of the way, let us get to dumping hoon functions!

We shall dump the most basic of functions, decrement!

We can do this simply by bringing decrement to the front of the environment, and getting it in the function form of [function sample environment] we saw before. We can do this by simply stating the name of the function we wish, and then get the function out of it by getting the second index!

Note that Hoon uses function:module (f:mn:...:m1) form.

.*  dec:anoma  [0 2]
[6 [5 [1 0] 0 6] [0 0] 8 [1 0] 8 [1 6 [5 [0 30] 4 0 6] [0 6] 9 2 10 [6 4 0 6] 0 1] 9 2 0 1]

The logic here doesn't particularly matter, but here we have the nock definiton of decrement, which is wonderful!

This can be done to any function, regardless of how nested the modules are

> .*  lsh:block:anoma  [0 2]
[ 8
  [9 4 0 255]
  9
  2
  10
  [6 [0 29] 7 [0 3] 8 [9 4 0 31] 9 2 10 [6 7 [0 3] 8 [9 4 0 255] 9 2 10 [6 [7 [0 3] 9 182 0 7] 0 28] 0 2] 0 2]
  0
  2
]

Casting to Nock, a useful tool

A good way to visualize the dump, is by casting the result to nock

> ;;  nock  [9 2 0 1]
[%9 p=2 q=[%0 p=1]

the p's and q's are arguments and the %9 and %0 are the nock instructions being ran.

From here, the instruction set can be consluted for the meaning of any particular instruction.

> ;;  nock  .*  dec.anoma  [0 2]
[ %6
  p=[%5 p=[%1 p=0] q=[%0 p=6]]
  q=[%0 p=0]
  r=[%8 p=[%1 p=0] q=[%8 p=[%1 p=[6 [5 [0 30] 4 0 6] [0 6] 9 2 10 [6 4 0 6] 0 1]] q=[%9 p=2 q=[%0 p=1]]]]
]

Dumping Types

Types in Hoon are just functions!

A good example can be found by looking at the resource-type

> .*  resource:resource-machine  [0 2]
[ 8
  [ [8 [7 [0 7] 9 47 0 1] 9 2 10 [6 0 28] 0 2]
    [6 [6 [3 0 26] [1 1] 1 0] [0 26] 0 0]
    [6 [6 [3 0 54] [1 1] 1 0] [0 54] 0 0]
    [6 [6 [3 0 110] [1 1] 1 0] [0 110] 0 0]
    [6 [5 [1 0] 0 222] [1 0] 6 [5 [1 1] 0 222] [1 1] 0 0]
    [6 [6 [3 0 446] [1 1] 1 0] [0 446] 0 0]
    [6 [6 [3 0 894] [1 1] 1 0] [0 894] 0 0]
    6
    [6 [3 0 895] [1 1] 1 0]
    [0 895]
    0
    0
  ]
  8
  [5 [0 14] 0 2]
  0
  6
]

This however does not show how to dump the structure of a type well enough, however this can be fixed by simply just calling it!

calling a type leads to something like this

> (resource:resource-machine)
[   logic
  < 1|xpg
    [ [ roots=it(@)
        commitments=it(@)
        nullifiers=it(@)
        proofs=it(#4)
        delta=it([denom=@ sign=?(%.y %.n) amount=@])
        extra=@
        preference=%~
      ]
      [ roots=it(@)
        commitments=it(@)
        nullifiers=it(@)
          proofs
        it( ^#4
          [   logic
            < 1|xpg
              [ [ roots=it(@)
                  commitments=it(@)
                  nullifiers=it(@)
                  proofs=it(#4)
                  delta=it([denom=@ sign=?(%.y %.n) amount=@])
                  extra=@
                  preference=%~
                ]
                [ roots=it(@)
                  commitments=it(@)
                  nullifiers=it(@)
                  proofs=it(#4)
                  delta=it([denom=@ sign=?(%.y %.n) amount=@])
                  extra=@
                  preference=%~
                ]
                ?(%.y %.n)
              ]
            >
            label=@t
            quantity=@
            data=@
            eph=?(%.y %.n)
            nonce=@
            npk=@
            rseed=@
          ]
        )
        delta=it([denom=@ sign=?(%.y %.n) amount=@])
        extra=@
        preference=%~
      ]
      ?(%.y %.n)
    ]
  >
  label=''
  quantity=0
  data=0
  eph=%.y
  nonce=0
  npk=0
  rseed=0
]

Which just gives the default values. If the result is hard to read, then no problem, just forget the type information!

> `*`(resource:resource-machine)
[[[0 15] [0 0 0 0 0 0 0] [0 0 0 0 0 0 0] 0] 0 0 0 0 0 0 0]

Here we simply jsut cast it to the any type, forgetting all information, and we can now see the format of the empty resource.

Dump Modules

Dumping modules is the same as dumping functions, it's just a matter that one's terminal will be flooded

> .*  resource-machine  [0 2]
[ [1 0]
...
  0
  1
]

Thus feel free to dump away. This is only useful when trying to copy this to the Elixir codebase.

Dumping Hoon for Elixir

Since Anoma itself runs Nock and not Hoon, we have to take the Hoon code we have and include it in Elixir somehow.

This process isn't particular difficult, and we can do it by simply using the tools we've learned in this document.

For example, it's not uncommon when the standard library that test indicies are not out of date and need to be updated, or maybe we define out a new hoon function for testing.

In these scenarios, there is a very easy way to update the code.

Let us look at the fibonacci example in Elixir

# can be found in https://github.com/anoma/anoma/blob/base/lib/test_helper/nock.ex
  @spec factorial() :: Noun.t()
  def factorial() do
    arm = Noun.Format.parse_always("
    [ 8
      [1 1 0]
      8
      [ 1
        6
        [5 [0 30] 1 0]
        [0 13]
        9
        2
        10
        [30 8 [9 342 0 255] 9 2 10 [6 0 62] 0 2]
        10
        [6 [8 [9 20 0 255] 9 2 10 [6 [0 29] 0 28] 0 2] 0 12]
        0
        1
      ]
      9
      2
      0
      1
    ]")
    sample = 1
    [arm, sample | logics_core()]
  end

Here we have some just plain old nock string representing the function, and we append the context to it via normal Elixir. We do this to save space, as we really don't want to dump the entire Nock.Lib.logics_core/0 for every simple function.

On the Hoon side we just run this to get the proper new logic

> .*  fib:tests  [0 2]
[ 8
  [1 1 0]
  8
  [ 1
    6
    [5 [0 30] 1 0]
    [0 13]
    9
    2
    10
    [30 8 [9 342 0 1.023] 9 2 10 [6 0 62] 0 2]
    10
    [6 [8 [9 20 0 1.023] 9 2 10 [6 [0 29] 0 28] 0 2] 0 12]
    0
    1
  ]
  9
  2
  0
  1
]

and then replace the old logic with the new code.

  @spec factorial() :: Noun.t()
  def factorial() do
    arm = Noun.Format.parse_always("
    [ 8
      [1 1 0]
      8
      [ 1
        6
        [5 [0 30] 1 0]
        [0 13]
        9
        2
        10
        [30 8 [9 342 0 1.023] 9 2 10 [6 0 62] 0 2]
        10
        [6 [8 [9 20 0 1.023] 9 2 10 [6 [0 29] 0 28] 0 2] 0 12]
        0
        1
      ]
      9
      2
      0
      1
    ]")
    sample = 1
    [arm,, sample | logics_core()]
  end

The process is the same for the code in Nock, just dump the [0 2] index of the module and replace the string with the result you get in your terminal.

Dumping Indexing Offsets

The tools that we have explored so far only deal with dumping definitions, however they do not explain where these functions are stored in the environment.

That is where zaptis(!=) comes handy.

zaptis(!=) simply gives us the hoon expression of the argument handed to it.

Let us start off simple with zaptis(!=), let us look at what saying anoma actually does.

> !=(anoma)
[0 46]

Interesting, we can see that saying anoma, indexs into the current environment by 46. The current environment in Hoon can be conjured with ..

> !=(.)
[0 1]

With this knowledge in hand, we can verify that anoma really is at index 46!

> =(.*(. [0 46]) anoma)
%.y

Now that we know how to get the index for names like anoma, what about trying to get the index of a function like dec inside of the anoma environment.

> !=(dec:anoma)
[7 [0 46] 9 342 0 15]
> ;;  nock  !=(dec:anoma)
[%7 p=[%0 p=46] q=[%9 p=342 q=[%0 p=15]]]

Here it's a bit more complicated to let us break it down step by step.

  1. [%7 p=[%0 p=46] q=...]
    • In the section where we are calling %7.
    • This has the effect of just trying to get anoma to be the subject of the following q computation.
  2. Now at q=[%9 p=342 q=[%0 p=15]] we are running this on anoma itself.
    • %9 is rather basic, trying to call the given index p at arm q.
    • In our case, dec is located at index 342 inside of arm at the layer/module located at [0 15].
    • [0 15] is really layer 1 in the source code and is properly documented as such
~%  %one  +  ~
|%
++  dec  ::  +342
  ~/  %dec
  |=  a=@
  ?<  =(0 a)
  =|  b=@
  |-  ^-  @
  ?:  =(a +(b))  b
  $(b +(b))

Thus, it's not very complicated, thus in the form

> !=(dec:anoma)
[7 [0 46] 9 342 0 15]

all we have to pay attention to is the 342 and the 15, some more examples show this off well

> !=(dec:anoma) ::  index 342 at layer 1
[7 [0 46] 9 342 0 15]
> !=(add:anoma) ::  index 20 at layer 1
[7 [0 46] 9 20 0 15]
> !=(trap:anoma) ::  index 20 at layer 2
[7 [0 46] 9 20 0 7]
> !=(unit:anoma) ::  index 42 at layer 2
[7 [0 46] 9 42 0 7]

Here for any non nested module we can see the layers and indexs quite plainly!

How Index of Layers Change

The hoon environment is a binary tree. Included below is an extended diagram that we will use for our explanation.

stateDiagram-v2
1 --> 2
1 --> 3
3 --> 6
3 --> 7
7 --> 14
7 --> 15
15 --> 30
15 --> 31
Loading

Whenever, a layer is made in hoon, we should think of it as pushed onto the env. So for Anoma the layers can be seen like this

stateDiagram-v2
layer_four --> code_in_layer_four
layer_four --> layer_three
layer_three --> code_in_layer_three
layer_three --> layer_two
layer_two --> code_in_layer_two
layer_two --> layer_one
layer_one --> code_in_layer_one
layer_one --> 0_3_99
Loading

If we pushed layer 5, then everything shifts, layer 1 moves from 15 to 31.

Thus the indexing works on a rather simple formula that can be read about: here. The code is not exactly this formula, but below we will show how it shapes up.

series = 1..5 |> Enum.map(fn i -> 2 ** i - 1 end)
indicies = 1..5
my_data = %{series: series, indicies: indicies}
%{series: [1, 3, 7, 15, 31], indicies: 1..5}
VegaLite.new(width: 200, height: 300, title: "Indexing Series")
|> VegaLite.data_from_values(my_data, only: ["indicies", "series"])
|> VegaLite.mark(:bar)
|> VegaLite.encode_field(:x, "indicies", type: :quantitative)
|> VegaLite.encode_field(:y, "series", type: :quantitative)