|
| 1 | +# Updating the contract |
| 2 | + |
| 3 | +[Earlier](./first_steps.md) we said that if an `i8` is currently `127` and you increment it again, it will _overflow_, becoming `-128`. Likewise, it can _underflow_ if currently `-128` and you decrement. |
| 4 | + |
| 5 | +Let's see if we can make `val` overflow! And then let's see how to prevent this. |
| 6 | + |
| 7 | +# Method 1: mash `increment` 128 times |
| 8 | + |
| 9 | +With the existing contract, you could use the RAEN Admin panel and mash the Submit button on the `increment` form 128 times. |
| 10 | + |
| 11 | +This could take a while, though! Let's see if we can improve the code to make it easier. |
| 12 | + |
| 13 | +# Method 2: new `default` |
| 14 | + |
| 15 | +If you're following along with your own contract, you can change the default value of `val` to start at `127` rather than `0`. This is a good way to learn about where the default of `0` currently comes from. |
| 16 | + |
| 17 | +If you get stuck while following along, you can see this method and the following all implemented on [this branch](https://github.com/raendev/examples/pull/4/files). |
| 18 | + |
| 19 | +Above `pub struct Counter`, change the `derive` macro by removing the `Default` trait: |
| 20 | + |
| 21 | +```diff |
| 22 | +-#[derive(Default, BorshDeserialize, BorshSerialize)] |
| 23 | ++#[derive(BorshDeserialize, BorshSerialize)] |
| 24 | +``` |
| 25 | + |
| 26 | +Now, below the closing bracket of `pub struct Counter`, you can implement `default` manually: |
| 27 | + |
| 28 | +```rust |
| 29 | +pub struct Counter { |
| 30 | + val: i8, |
| 31 | +} |
| 32 | + |
| 33 | +impl Default for Counter { |
| 34 | + fn default() -> Self { |
| 35 | + Self { val: 127 } |
| 36 | + } |
| 37 | +} |
| 38 | +``` |
| 39 | + |
| 40 | +## If you're following along |
| 41 | + |
| 42 | +If you're following along with your own code, you can build and re-deploy to see the new default: |
| 43 | + |
| 44 | +```bash |
| 45 | +raen build --release -q | xargs -I {} near dev-deploy --wasmFile {} |
| 46 | +``` |
| 47 | + |
| 48 | +(This [uses xargs to](https://stackoverflow.com/a/42631216/249801) pipe the output of `raen build` to the `wasmFile` argument of `near dev-deploy`.) |
| 49 | + |
| 50 | +If you've already incremented or decremented the number, though, you'll see that this made no change. Which is what you'd expect! It only changes the _default_. That is, the value that is shown if you didn't actually change it and store the new value. |
| 51 | + |
| 52 | +To see this default in action, you can delete the `neardev` folder and run the `dev-deploy` command again. This will create a brand new NEAR testnet account before deploying your contract. |
| 53 | + |
| 54 | +Paste this new account name into RAEN Admin, check `get_num`, and see the number you set as a default! Call `increment` (you'll need to sign in again, since this is a new contract.) |
| 55 | + |
| 56 | +## If you're not |
| 57 | + |
| 58 | +The contract deployed at [counter.raendev.testnet](https://raen.dev/admin/#/counter.raendev.testnet) already has a value for `val` saved to contract storage. It will not make use of a new default. |
| 59 | + |
| 60 | +Let's try a different way. |
| 61 | + |
| 62 | +# Method 3: new `set_num` |
| 63 | + |
| 64 | +See if you can write your own `set_num` method based on the code for `increment` and `decrement` in the contract. If you need a hint, [here's what it looks like](https://github.com/raendev/examples/pull/4/files). |
| 65 | + |
| 66 | +## If you're following along |
| 67 | + |
| 68 | +Save your file with new `set_num`, then build and re-deploy: |
| 69 | + |
| 70 | +```bash |
| 71 | +raen build --release -q | xargs -I {} near dev-deploy --wasmFile {} |
| 72 | +``` |
| 73 | + |
| 74 | +Unlike Method 2, you don't need to create a new testnet account. Just refresh the RAEN Admin panel and see the `set_num` method appear under Change Methods. Go ahead and try it out! |
| 75 | + |
| 76 | +## If you're not |
| 77 | + |
| 78 | +The [reference code](https://github.com/raendev/examples/pull/4/files) has been deployed to [v2.counter.raendev.testnet](https://raen.dev/admin/#/v2.counter.raendev.testnet) for you to play with. |
| 79 | + |
| 80 | +Note that this code already fixed the overflow issue. If you want to see the overflow in action, you'll have to follow along with your own code. |
| 81 | + |
| 82 | +# Fixing that overflow |
| 83 | + |
| 84 | +Now you can call `increment` and see that the number goes from `127` to `-128`. Oops! How do we fix it? |
| 85 | + |
| 86 | +Open up the `Cargo.toml` file. [Cargo.toml](https://doc.rust-lang.org/cargo/guide/cargo-toml-vs-cargo-lock.html) is to Rust what `package.json` is to NodeJS. You'll see sections for `dependencies` and `dev-dependencies` and more generic information about this package/contract/crate. |
| 87 | + |
| 88 | +See those lines with `overflow-checks = false`? Change them to `overflow-checks = true`. Now rebuild and redeploy: |
| 89 | + |
| 90 | +```bash |
| 91 | +raen build --release -q | xargs -I {} near dev-deploy --wasmFile {} |
| 92 | +``` |
| 93 | + |
| 94 | +You'll see that this takes longer to build than when you make changes to your contract itself. Rust needs to recompile all of your dependencies using the new setting. |
| 95 | + |
| 96 | +Refresh your admin panel, set the number back to `127`, and call `increment`. You'll see an error message. This means the contract panicked rather than allowing the integer overflow. It worked! |
0 commit comments