Skip to content

Commit

Permalink
Merge pull request #42 from TomAFrench/frontend
Browse files Browse the repository at this point in the history
Frontend update
  • Loading branch information
TomAFrench authored Apr 11, 2020
2 parents 5a62ac7 + 839ae7c commit 52e402c
Show file tree
Hide file tree
Showing 32 changed files with 3,385 additions and 13,788 deletions.
5 changes: 5 additions & 0 deletions .gitbook.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
root: ./docs/

​structure:
readme: README.md
summary: SUMMARY.md
2 changes: 1 addition & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ build

# misc
.DS_Store
.env*
.env.*.local

# debug
npm-debug.log*
Expand Down
15 changes: 15 additions & 0 deletions docs/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
# Welcome to the NoteStream Docs

If you are interested in how to use NoteStream to privately stream Ethereum tokens, then please look at our [Getting Started](./getting_started/README.md) page.

# Github

The codebase for the NoteStream contracts and frontend are hosted publicly on [Github](https://github.com/TomAFrench/NoteStream).

# Networks

As NoteStream is in rapid development, it is currently only deployed on the Rinkeby test network. We hope to deploy to other testnets and subsequently the Ethereum Mainnet in due time.

# Gas Costs

We have not measured the gas costs associated with using NoteStream in detail, however we expect the majority of the cost to be from validating the ZK proof required for all transactions using AZTEC Protocol. Since EIP 1108, this takes in the range of 200,000-300,000 gas which corresponds to roughly 5-6x that of a standard ERC20 token transfer.
20 changes: 20 additions & 0 deletions docs/SUMMARY.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# Summary​


* [Getting Started](getting_started/README.md)
* [Using Aztec notes](using_aztec/README.md)
* [Comparison to other money streaming options](comparison_to_others/README.md)

<!-- ## Use headings to create page groups like this one​
* [First page's title](page1/README.md)
* [Some child page](page1/page1-1.md)
* [Some other child page](part1/page1-2.md)
* [Second page's title](page2/README.md)
* [Some child page](page2/page2-1.md)
* [Some other child page](part2/page2-2.md)
## A second-page group​
* [Yet another page](another-page.md) -->
1 change: 1 addition & 0 deletions docs/comparison_to_others/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
# Sablier
89 changes: 89 additions & 0 deletions docs/getting_started/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
# Getting Started
## Trade ERC20 tokens for ZkAssets

Many of the tokens which you will want to stream are going to be ERC20 tokens such as DAI. These don't have the privacy features needed for NoteStream to work. Luckily we can wrap these tokens into an ZkAsset before we start streaming them.
To do this click the "Deposit ERC20 tokens for private assets" and enter the number of tokens you want to convert into ZkAssets.

## Deposit a note into a stream

You can create a stream by clicking the "Create a new stream" button and then following the shown instructions.
It is important to note that the stream recipient must have registered for an AZTEC address in order for then to be able to receive a stream. This is done automatically upon first visiting the NoteStream website.

## Withdrawing from a stream

The NoteStream app continually calculates the maximum amount which you can withdraw from the streams you are receiving. If you click the "withdraw" button next to the stream you want to withdraw from then a ZK proof will be generated to withdraw this amount from the stream note to your wallet.

## Converting back into ERC20 tokens

Once you have withdrawn your ZkAssets, you can convert them back into ERC20 tokens in much the same way as converting ERC20s into ZkAssets.
Of course, when you convert ZkAssets back into ERC20 tokens you lose the privacy properties given by AZTEC notes. There is also the possibility to leak information on the value of a stream if you immediately convert a withdrawal back into ERC20 tokens.

# Terminology
## AZTEC Note

AZTEC notes can be thought of as tokens on Ethereum for which all balances are encrypted so only their owners can view them.
This is a gross oversimplification as these notes instead work on UTXO model (similar to Bitcoin) rather than then fungible balance model of ERC20 tokens which are all familiar with.

## ZkAsset

A zero knowledge representation of an ERC20 token. For example: DAI is represented by the zkDAI ZkAsset.

## Stream Note
The AZTEC note held by the NoteStream contract for a given stream. This note represents the total value of the stream at any time.

# How it works
## Where is my money held?

Each stream is made up of an AZTEC note locked on the NoteStream smart contract. The logic of this smart contract is such that only the stream's sender or receiver may interact with this note.

## How can my streams be private if everything on Ethereum is public?

You're right that everything that happens on the Ethereum network is available for anyone to inspect, however NoteStream uses AZTEC Protocol which allows funds to be transferred as "notes" for which the value is encrypted. Everyone can see that a stream exists but nobody but you will know how much value it contains.

# Interacting with NoteStream
## Can I cancel streams?

Yes. Both the stream sender and recipient can cancel the stream at any time. This will send the appropriate fraction of the stream note's value to each party and then delete the stream.
Can I modify a stream in progress?
No. We're looking at the possibility to allow a stream's sender to modify a stream in progress in certain ways, e.g. extending the stream by topping up the stream note's value.

# Privacy

There are a number of privacy enhancing measures you can take using ZkAssets which are general rather than NoteStream-specific. Please see ["Using Aztec notes"](./using_aztec/README.md) for more information.

## NoteStream-specific public information

There are two times at which information about the stream is made visible

### Creating a stream

A stream is made up of the following information:
- Sender address
- Recipent address
- Stream note hash
- ZkAsset address
- Start Time
- Stop Time

An observer will then be able to see who is streaming to whom, what kind of token they are streaming and when this stream occurred. e.g.
> Alice streamed Bob an unknown amount of zkDAI represented by the AZTEC note with hash 0x1a3...cE1 from 9:00am until 5:00pm on 12/4/20
### Withdrawing/cancelling a stream

Withdrawal and cancellation transactions leak the same information. Here we discuss a withdrawal transaction as an example.

Each withdrawal transaction includes information on the fraction of the stream's duration which is being withdrawn. This is required such that the NoteStream contract can ensure that the withdrawal is valid.
An observer may then for example see that the recipient is withdrawing a value corresponding to a certain fraction of the remaining value on the stream note.

However it is important to note that without knowledge of the initial value of the stream note then it is impossible to determine the absolute value being withdrawn. e.g.

> Bob withdrew 50% of the value of the stream at 1:00pm on 12/4/20

# Security
## Is NoteStream safe?
Currently there are a number of known security flaws which mean that NoteStream should not be used for any Mainnet funds (as such, there is no current Mainnet deployment.). I'm currently speaking with AZTEC about updates to their SDK in order to fix these.

## How do I know you can't steal my funds?

All Notestream contracts are open source and verified on Etherscan.
20 changes: 20 additions & 0 deletions docs/using_aztec/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# Issues associated with depositing/withdrawing ERC20 tokens

An important factor to keep in mind is that the process of depositing ERC20s into a ZkAsset doesn't immediately make them private, i.e. there will be a transaction visible on the blockchain in which a certain amount of DAI is converted into a number of zkDAI notes.
It's impossible to tell what each individual note is worth but the sum of them must equal the number of ERC20s deposited. In the worst case scenario, if all of those notes are then used in a single transaction (such as creation of a NoteStream stream) then it's obvious that the transaction value is equal to that of the ERC20 deposit.

This might sound like it means that it's impossible to have privacy using ZkAssets as anyone can trace your notes back to when they were deposited. However as people send ZkAssets to each other and notes are split and joined, a given deposit may be linked to a huge amount of notes spread over a vast number of people. We're very quickly at a point where we can see that 1000 people all together own the value from a given deposit but it's impossible to work out exactly who owns what fraction.

## Improving privacy of deposits

Even before this mixing behaviour there are steps you can take to improve your privacy. When depositing ERC20s into a ZkAsset its possible to create a number of notes which have zero value attached. This might sound pointless but it allows you then spend your entire deposit without letting anyone know how much you've spent.

An observer will only be able to tell that your stream is worth at most equal to your deposit but it could be anything less than that.
This behaviour is implemented automatically by the AZTEC sdk so you don't need to worry about it.

## Take aways

In order to improve the privacy of your transactions using ZkAssets it is best to

- Have a long history of transactions using this ZkAsset since your last deposit (idealling receiving ZkAsset funds from other people as well)
- Deposit an amount of ERC20 tokens in excess of what you are planning on immediately streaming.
9 changes: 9 additions & 0 deletions now.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
{
"version": 2,
"build": {
"env": {
"REACT_APP_SUBGRAPH_URL": "https://api.thegraph.com/subgraphs/name/tomafrench/notestream-rinkeby",
"REACT_APP_NETWORK_ID": "4"
}
}
}
6 changes: 4 additions & 2 deletions packages/contract-artifacts/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -13,11 +13,13 @@
"@babel/plugin-proposal-object-rest-spread": "^7.4.0",
"@babel/preset-env": "^7.9.0",
"@babel/preset-typescript": "^7.9.0",
"@typescript-eslint/eslint-plugin-tslint": "^2.27.0",
"eslint": "^5.15.3",
"eslint-config-airbnb-base": "^13.1.0",
"eslint-config-prettier": "^6.0.0",
"eslint-plugin-import": "^2.16.0",
"shx": "^0.3.2"
"eslint-plugin-import": "^2.20.2",
"shx": "^0.3.2",
"typescript": "^3.8.3"
},
"engines": {
"node": ">=8.3"
Expand Down
2 changes: 2 additions & 0 deletions packages/react-app/.env
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
REACT_APP_SUBGRAPH_URL='https://api.thegraph.com/subgraphs/name/tomafrench/notestream-rinkeby'
REACT_APP_NETWORK_ID=4
5 changes: 3 additions & 2 deletions packages/react-app/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -17,17 +17,18 @@
"@material-ui/icons": "^4.9.1",
"@notestream/contract-artifacts": "^1.0.1",
"aztec.js": "^0.17.0",
"bignumber.js": "^9.0.0",
"bnc-onboard": "^1.7.1",
"ethers": "^4.0.46",
"graphql": "^14.6.0",
"moment": "^2.24.0",
"prop-types": "^15.7.2",
"react": "^16.13.0",
"react-blockies": "^1.4.1",
"react-copy-to-clipboard": "^5.0.2",
"react-dom": "^16.13.0",
"react-router": "^5.1.2",
"react-router-dom": "^5.1.2",
"react-scripts": "3.4.0",
"web3": "^1.2.6",
"zkasset-metadata": "^0.1.2"
},
"devDependencies": {
Expand Down
81 changes: 29 additions & 52 deletions packages/react-app/src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,25 +7,18 @@ import AppBar from '@material-ui/core/AppBar';
import Tabs from '@material-ui/core/Tabs';
import Tab from '@material-ui/core/Tab';
import Box from '@material-ui/core/Box';
import Toolbar from '@material-ui/core/Toolbar';
import LocalAtmIcon from '@material-ui/icons/LocalAtm';

import getZkAssetsForNetwork from 'zkasset-metadata';
import { getContractAddressesForNetwork, abis } from '@notestream/contract-artifacts';
import { ApolloClient, ApolloProvider, HttpLink, InMemoryCache } from '@apollo/client';
import Web3 from 'web3';
import { ethers } from 'ethers';

import Status from './components/status';
import getWeb3 from './utils/web3';
import DepositDialog from './components/modals/DepositModal';
import WithdrawDialog from './components/modals/WithdrawModal';
import CreateStreamDialog from './components/modals/CreateStreamModal';

const client = new ApolloClient({
cache: new InMemoryCache(),
link: new HttpLink({
uri: 'https://api.thegraph.com/subgraphs/name/tomafrench/notestream-rinkeby',
}),
});
import { useAddress, useWallet } from './contexts/OnboardContext';
import setupAztec from './utils/setup';
import Header from './components/header/Header';

const useStyles = makeStyles((theme) => ({
layout: {
Expand Down Expand Up @@ -73,69 +66,53 @@ function TabPanel(props: any): ReactElement {
);
}

const NETWORK_ID: number = parseInt(process.env.REACT_APP_NETWORK_ID as string, 10);

const App = (): ReactElement => {
const classes = useStyles();
const [userAddress, setUserAddress] = useState('');
const userAddress = useAddress();
const wallet = useWallet();
const [streamContractInstance, setStreamContractInstance] = useState({});
const [value, setValue] = useState(0);

const addresses = getContractAddressesForNetwork(4);
const zkAssets = getZkAssetsForNetwork(4);
const [aztec, setAztec] = useState({} as any);
const addresses = getContractAddressesForNetwork(NETWORK_ID);
const zkAssets = getZkAssetsForNetwork(NETWORK_ID);

useEffect(() => {
async function initialiseAztec(): Promise<void> {
const account = await window.aztec.enable({
contractAddresses: {
ACE: addresses.ACE,
},
apiKey: '9HRKN7S-JSZMRJM-KWSDWSY-B2VSRD9', // API key for use with GSN for free txs.
window.addEventListener('load', () => {
setupAztec(NETWORK_ID).then(() => {
setAztec(window.aztec);
});
if (account) {
console.log('Initialised AZTEC');
}
}
initialiseAztec();
}, [userAddress, addresses]);
});
}, []);

useEffect(() => {
async function init(): Promise<void> {
const web3: Web3 = await getWeb3();
const accounts = await web3.eth.getAccounts();
setUserAddress(accounts[0]);

const streamContract = new web3.eth.Contract(abis.NoteStream, addresses.NoteStream);
if (wallet.provider) {
const signer = new ethers.providers.Web3Provider(wallet.provider).getSigner();
const streamContract = new ethers.Contract(addresses.NoteStream, abis.NoteStream, signer);
setStreamContractInstance(streamContract);
}
init();
}, [addresses.NoteStream]);
}, [wallet.provider, addresses.NoteStream]);

return (
<ApolloProvider client={client}>
<AppBar position="static">
<Toolbar>
<LocalAtmIcon className={classes.icon} />
<Typography variant="h6" className={classes.title}>
NoteStream
</Typography>
{/* <Button className={classes.button} variant="contained" >Connect to Wallet</Button> */}
</Toolbar>
</AppBar>
<>
<Header />
<main className={classes.layout}>
<Paper className={`${classes.pageElement} ${classes.paper}`}>
<Grid container direction="row" justify="space-around" spacing={3}>
<Grid item>
<DepositDialog aztec={window.aztec} zkAssets={zkAssets} userAddress={userAddress} />
<DepositDialog aztec={aztec} zkAssets={zkAssets} userAddress={userAddress} />
</Grid>
<Grid item>
<CreateStreamDialog
aztec={window.aztec}
aztec={aztec}
zkAssets={zkAssets}
userAddress={userAddress}
streamContractInstance={streamContractInstance}
/>
</Grid>
<Grid item>
<WithdrawDialog aztec={window.aztec} zkAssets={zkAssets} userAddress={userAddress} />
<WithdrawDialog aztec={aztec} zkAssets={zkAssets} userAddress={userAddress} />
</Grid>
</Grid>
</Paper>
Expand All @@ -151,22 +128,22 @@ const App = (): ReactElement => {
<Status
role="sender"
userAddress={userAddress}
aztec={window.aztec}
aztec={aztec}
streamContractInstance={streamContractInstance}
/>
</TabPanel>
<TabPanel value={value} index={1}>
<Status
role="recipient"
userAddress={userAddress}
aztec={window.aztec}
aztec={aztec}
streamContractInstance={streamContractInstance}
/>
</TabPanel>
</Paper>
</Grid>
</main>
</ApolloProvider>
</>
);
};

Expand Down
Loading

1 comment on commit 52e402c

@vercel
Copy link

@vercel vercel bot commented on 52e402c Apr 11, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please sign in to comment.