Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add logger, deno tasks, and README to examples #20

Merged
merged 7 commits into from
Aug 4, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 9 additions & 1 deletion cli/start/mod.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ import { ArkiverMetadata } from '../../src/arkiver/arkive-metadata.ts'
import { createManifestHandlers } from './logger.ts'
import { colors, mongoose, SchemaComposer } from '../../src/deps.ts'
import { logger } from '../../src/logger.ts'
import { collectRpcUrls } from '../utils.ts'


export const action = async (
options: {
Expand Down Expand Up @@ -114,11 +116,17 @@ export const action = async (
},
})

// An RPC for our Arkive is going to be assigned at some point.
// The order of assignment is as follows:
// 1. CLI command line option -r, --rpc-url
// 2. Env variables such as {CHAIN}_RPC_URL
// 3. RPC url defined in manifest
// 4. Default RPC of Viem
const rpcUrls = options.rpcUrl?.reduce((acc, rpc) => {
const [name, url] = rpc.split('=')
acc[name] = url
return acc
}, {} as Record<string, string>) ?? {}
}, {} as Record<string, string>) ?? collectRpcUrls() ?? {}

logger('arkiver').debug(`Connecting to database...`)
const connectionString = options.mongoConnection ??
Expand Down
21 changes: 21 additions & 0 deletions cli/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { SUPABASE_ANON_PUBLIC_KEY, SUPABASE_URL } from './constants.ts'
import { createClient, Input, Secret, z } from './deps.ts'
import { login } from './login/mod.ts'
import { spinner } from './spinner.ts'
import { supportedChains } from '../src/chains.ts'

export const getEmail = async () => {
const email = await Input.prompt('✉️ Email:')
Expand Down Expand Up @@ -127,3 +128,23 @@ export const craftEndpoint = (
majorVersion ? majorVersion + '/' : ''
}graphql`
}

export const getEnv = (key: string, defaultValue?: string): string => {
const value = Deno.env.get(key)
if (!value && !defaultValue) {
throw new Error(`Missing environment variable: ${key}`)
}
return value || defaultValue || ''
}

export const collectRpcUrls = () => {
const rpcUrls: Record<string, string> = {}
for (const chain of Object.keys(supportedChains)) {
try {
rpcUrls[chain] = getEnv(`${chain.toUpperCase()}_RPC_URL`)
} catch (_e) {
// ignore
}
}
return rpcUrls
}
49 changes: 45 additions & 4 deletions deno.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

38 changes: 38 additions & 0 deletions examples/block-handler-vaults/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
# ERC20 Balance History
This Arkive keeps an accurate historical record of the sharePrice of a given set of YearnV2 Abi Vaults. sharePrice increases as profits are distributed to the vaults, but may also decrease if loss is incurred. It may be valuable to track this information in case of incidents or in order to optimize the vaults for instance.
### Dependencies
* Docker
* Full Archive RPC (Infura, Ankr, Alchemy, etc)

### Arkive Usage

First make sure .env is configured correctly with your RPC endpoint. In this example we are connecting the `mainnet` with the Ankr public ETH endpoint.
> MAINNET_RPC_URL=https://rpc.ankr.com/eth

All available tasks can been seen with
> deno task

To start a new instance of the Arkive you can use `deno task` to run the `new` script with the following command
> deno task new

To reset the database and resync the Arkive run `reset` task
> deno task reset

### Using the GraphQL Explorer
If Arkiver is running there should now be webpage avaiable at http://0.0.0.0:4000/graphql

In the left side you can use the no-code explorer to pick and choose which Entities and what fields of the Entities you would like to look at. Here is an example query for this Arkive. This fetches the latest VaultSnapshot entity.
```
query MyQuery {
VaultSnapshot(sort: _ID_DESC) {
vault
name
symbol
block
timestamp
sharePrice
}
}
```


2 changes: 1 addition & 1 deletion examples/block-handler-vaults/abis/YearnV2.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
export const YearnV2Abi = [{
export const YEARN_V2_ABI = [{
'name': 'Transfer',
'inputs': [{ 'name': 'sender', 'type': 'address', 'indexed': true }, {
'name': 'receiver',
Expand Down
13 changes: 13 additions & 0 deletions examples/block-handler-vaults/deno.jsonc
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
{
"tasks": {
"start": "arkiver start ./ -c mongodb://localhost:27017",
"new": "docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest && arkiver start ./ -c mongodb://localhost:27017",
"resetdb": "docker stop mongo && docker rm mongo && docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest",
"reset": "docker stop mongo && docker rm mongo && docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest && arkiver start ./ -c mongodb://localhost:27017",
"newdb": "docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest",
"stopdb": "docker stop mongo",
"rmdb": "docker rm mongo",
"startdb": "docker start mongo",
"deploy": "arkiver deploy ./"
}
}
19 changes: 11 additions & 8 deletions examples/block-handler-vaults/handlers/vault.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { formatUnits, getContract } from 'npm:viem'
import { type BlockHandler } from 'https://deno.land/x/robo_arkiver@v0.4.19/mod.ts'
import { VaultSnapshot } from '../entities/vault.ts'
import { YearnV2Abi } from '../abis/YearnV2.ts'
import { YEARN_V2_ABI } from '../abis/YearnV2.ts'

const VAULTS = [
{ address: '0xdA816459F1AB5631232FE5e97a05BBBb94970c95', block: 12796965 }, // yvDAI
Expand All @@ -12,6 +12,7 @@ export const snapshotVault: BlockHandler = async ({
block,
client,
store,
logger,
}): Promise<void> => {
// Filter out vaults that haven't been deployed yet
const liveVaults = VAULTS.filter((e) => e.block < Number(block.number))
Expand All @@ -20,12 +21,12 @@ export const snapshotVault: BlockHandler = async ({
const vaults = await Promise.all(liveVaults.map(async (vault) => {
const contract = getContract({
address: vault.address,
abi: YearnV2Abi,
abi: YEARN_V2_ABI,
publicClient: client,
})
return {
address: vault.address,
vault: { address: vault.address, abi: YearnV2Abi } as const,
vault: { address: vault.address, abi: YEARN_V2_ABI } as const,
contract,
name: await store.retrieve(
`${vault.address}:name`,
Expand All @@ -46,22 +47,24 @@ export const snapshotVault: BlockHandler = async ({
const sharePrices = await Promise.all(vaults.map((e) => {
return client.readContract({
address: e.address,
abi: YearnV2Abi,
abi: YEARN_V2_ABI,
functionName: 'pricePerShare',
blockNumber: block.number,
})
}))

// Save the vault snapshots
vaults.map((vault, i) => {
const sharePrice = parseFloat(
formatUnits(sharePrices[i], Number(vault.decimals)),
)
logger.info(`${vault.name} share price updated to ${sharePrice}`)
return new VaultSnapshot({
id: `${vault.address}-${Number(block.number)}`,
// id: `${vault.address}-${Number(block.number)}`,
block: Number(block.number),
timestamp: Number(block.timestamp),
vault: vault.address,
sharePrice: parseFloat(
formatUnits(sharePrices[i], Number(vault.decimals)),
),
sharePrice: sharePrice,
name: vault.name,
symbol: vault.symbol,
})
Expand Down
13 changes: 7 additions & 6 deletions examples/block-handler-vaults/manifest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,13 @@ const manifest = new Manifest('yearn-vaults')

manifest
.addEntity(VaultSnapshot)
.addChain('mainnet')
.addBlockHandler({
blockInterval: 1000,
startBlockHeight: 12790000n,
handler: snapshotVault,
})
.addChain('mainnet', (chain) =>
chain
.addBlockHandler({
blockInterval: 1000,
startBlockHeight: 12790000n,
handler: snapshotVault,
}))

export default manifest
.build()
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
export default [{
export const ERC_20_ABI = [{
'inputs': [],
'stateMutability': 'nonpayable',
'type': 'constructor',
Expand Down
41 changes: 41 additions & 0 deletions examples/erc20-balance-history/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
# ERC20 Balance History
This Arkive keeps an accurate historical record of the balance of an account of a given ERC20 token. It uses the Balance entity as a mutable variable updates as events are handled and is used to keep track of an accounts balance as we sync eventually reflecting true onchain balance. BalanceHistory is an unchanging entity which stores balance over time and is indexed by the `block` field.
### Dependencies
* Docker
* Full Archive RPC (Infura, Ankr, Alchemy, etc)

### Arkive Usage

First make sure .env is configured correctly with your RPC endpoint. In this example we are connecting the `mainnet` with the Ankr public ETH endpoint.
> MAINNET_RPC_URL=https://rpc.ankr.com/eth

All available tasks can been seen with
> deno task

To start a new instance of the Arkive you can use `deno task` to run the `new` script with the following command
> deno task new

To reset the database and resync the Arkive run `reset` task
> deno task reset

### Using the GraphQL Explorer
If Arkiver is running there should now be webpage avaiable at http://0.0.0.0:4000/graphql

In the left side you can use the no-code explorer to pick and chose which Entities and what fields of the Entities you would like to look at. Here is an example query for this Arkive. This fetches the latest Balance and BalanceHistory entities.
```
query MyQuery {
Balance(sort: _ID_DESC) {
token
user
balance
}
BalanceHistory(sort: _ID_DESC) {
token
block
user
balance
}
}
```


13 changes: 13 additions & 0 deletions examples/erc20-balance-history/deno.jsonc
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
{
"tasks": {
"start": "arkiver start ./ -c mongodb://localhost:27017",
"new": "docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest && arkiver start ./ -c mongodb://localhost:27017",
"resetdb": "docker stop mongo && docker rm mongo && docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest",
"reset": "docker stop mongo && docker rm mongo && docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest && arkiver start ./ -c mongodb://localhost:27017",
"newdb": "docker run --name mongo -d -p 27017:27017 mongodb/mongodb-community-server:latest",
"stopdb": "docker stop mongo",
"rmdb": "docker rm mongo",
"startdb": "docker start mongo",
"deploy": "arkiver deploy ./"
}
}
4 changes: 2 additions & 2 deletions examples/erc20-balance-history/entities.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,15 +16,15 @@ export const Transfer = createEntity('Transfer', {
export const Balance = createEntity('Balance', {
token: String,
user: String,
balance: Number,
balance: String,
})

// Contains all balance changes for every user
export const BalanceHistory = createEntity('BalanceHistory', {
token: String,
block: { type: Number, index: true },
user: String,
balance: Number,
balance: String,
})

export const Entities = [Balance, BalanceHistory, Transfer]
Loading