diff --git a/apps/block_scout_web/lib/block_scout_web/controllers/chain_controller.ex b/apps/block_scout_web/lib/block_scout_web/controllers/chain_controller.ex index 48d84dd5d197..40a0fd3bbb68 100644 --- a/apps/block_scout_web/lib/block_scout_web/controllers/chain_controller.ex +++ b/apps/block_scout_web/lib/block_scout_web/controllers/chain_controller.ex @@ -4,6 +4,7 @@ defmodule BlockScoutWeb.ChainController do alias BlockScoutWeb.ChainView alias Explorer.{Chain, PagingOptions, Repo} alias Explorer.Chain.{Address, Block, Transaction} + alias Explorer.Chain.Supply.RSK alias Explorer.Counters.AverageBlockTime alias Explorer.ExchangeRates.Token alias Explorer.Market @@ -13,6 +14,15 @@ defmodule BlockScoutWeb.ChainController do transaction_estimated_count = Chain.transaction_estimated_count() block_count = Chain.block_estimated_count() + market_cap_calculation = + case Application.get_env(:explorer, :supply) do + RSK -> + RSK + + _ -> + :standard + end + exchange_rate = Market.get_exchange_rate(Explorer.coin()) || Token.null() render( @@ -22,6 +32,7 @@ defmodule BlockScoutWeb.ChainController do average_block_time: AverageBlockTime.average_block_time(), exchange_rate: exchange_rate, chart_data_path: market_history_chart_path(conn, :show), + market_cap_calculation: market_cap_calculation, transaction_estimated_count: transaction_estimated_count, transactions_path: recent_transactions_path(conn, :index), block_count: block_count diff --git a/apps/block_scout_web/lib/block_scout_web/templates/chain/show.html.eex b/apps/block_scout_web/lib/block_scout_web/templates/chain/show.html.eex index b5d7283a031c..17bd9c4afefe 100644 --- a/apps/block_scout_web/lib/block_scout_web/templates/chain/show.html.eex +++ b/apps/block_scout_web/lib/block_scout_web/templates/chain/show.html.eex @@ -30,7 +30,7 @@ <%= gettext "Market Cap" %> - + diff --git a/apps/block_scout_web/lib/block_scout_web/views/chain_view.ex b/apps/block_scout_web/lib/block_scout_web/views/chain_view.ex index 7f49d2d239ea..506d6ff3c879 100644 --- a/apps/block_scout_web/lib/block_scout_web/views/chain_view.ex +++ b/apps/block_scout_web/lib/block_scout_web/views/chain_view.ex @@ -2,4 +2,12 @@ defmodule BlockScoutWeb.ChainView do use BlockScoutWeb, :view alias BlockScoutWeb.LayoutView + + defp market_cap(:standard, exchange_rate) do + exchange_rate.market_cap_usd + end + + defp market_cap(module, exchange_rate) do + module.market_cap(exchange_rate) + end end diff --git a/apps/explorer/config/config.exs b/apps/explorer/config/config.exs index 9a7d9f42963d..9a601e6c688f 100644 --- a/apps/explorer/config/config.exs +++ b/apps/explorer/config/config.exs @@ -71,8 +71,15 @@ else config :explorer, Explorer.Staking.EpochCounter, enabled: false end -if System.get_env("SUPPLY_MODULE") == "TokenBridge" do - config :explorer, supply: Explorer.Chain.Supply.TokenBridge +case System.get_env("SUPPLY_MODULE") do + "TokenBridge" -> + config :explorer, supply: Explorer.Chain.Supply.TokenBridge + + "rsk" -> + config :explorer, supply: Explorer.Chain.Supply.RSK + + _ -> + :ok end if System.get_env("SOURCE_MODULE") == "TransactionAndLog" do diff --git a/apps/explorer/lib/explorer/chain/supply/rsk.ex b/apps/explorer/lib/explorer/chain/supply/rsk.ex new file mode 100644 index 000000000000..09a02252a289 --- /dev/null +++ b/apps/explorer/lib/explorer/chain/supply/rsk.ex @@ -0,0 +1,103 @@ +defmodule Explorer.Chain.Supply.RSK do + @moduledoc """ + Defines the supply API for calculating supply for coins from RSK. + """ + + use Explorer.Chain.Supply + + import Ecto.Query, only: [from: 2] + + alias Explorer.Chain.Address.CoinBalance + alias Explorer.Chain.{Block, Wei} + alias Explorer.ExchangeRates.Token + alias Explorer.{Market, Repo} + + def market_cap(exchange_rate) do + circulating() * exchange_rate.usd_value + end + + @doc "Equivalent to getting the circulating value " + def supply_for_days(days) do + now = Timex.now() + + balances_query = + from(balance in CoinBalance, + join: block in Block, + on: block.number == balance.block_number, + where: block.consensus == true, + where: balance.address_hash == ^"0x0000000000000000000000000000000001000006", + where: block.timestamp > ^Timex.shift(now, days: -days), + distinct: fragment("date_trunc('day', ?)", block.timestamp), + select: {block.timestamp, balance.value} + ) + + balance_before_query = + from(balance in CoinBalance, + join: block in Block, + on: block.number == balance.block_number, + where: block.consensus == true, + where: balance.address_hash == ^"0x0000000000000000000000000000000001000006", + where: block.timestamp <= ^Timex.shift(Timex.now(), days: -days), + order_by: [desc: block.timestamp], + limit: 1, + select: balance.value + ) + + by_day = + balances_query + |> Repo.all() + |> Enum.into(%{}, fn {timestamp, value} -> + {Timex.to_date(timestamp), value} + end) + + starting = Repo.one(balance_before_query) || wei!(0) + + result = + -days..0 + |> Enum.reduce({%{}, starting.value}, fn i, {days, last} -> + date = + now + |> Timex.shift(days: i) + |> Timex.to_date() + + case Map.get(by_day, date) do + nil -> + {Map.put(days, date, last), last} + + value -> + {Map.put(days, date, value.value), value.value} + end + end) + |> elem(0) + + {:ok, result} + end + + def circulating do + query = + from(balance in CoinBalance, + join: block in Block, + on: block.number == balance.block_number, + where: block.consensus == true, + where: balance.address_hash == ^"0x0000000000000000000000000000000001000006", + order_by: [desc: block.timestamp], + limit: 1, + select: balance.value + ) + + Repo.one(query) || wei!(0) + end + + defp wei!(value) do + {:ok, wei} = Wei.cast(value) + wei + end + + def total do + 21_000_000 + end + + def exchange_rate do + Market.get_exchange_rate(Explorer.coin()) || Token.null() + end +end diff --git a/apps/explorer/test/explorer/chain/supply/rsk_test.exs b/apps/explorer/test/explorer/chain/supply/rsk_test.exs new file mode 100644 index 000000000000..40aa0a831b8e --- /dev/null +++ b/apps/explorer/test/explorer/chain/supply/rsk_test.exs @@ -0,0 +1,139 @@ +defmodule Explorer.Chain.Supply.RSKTest do + use Explorer.DataCase + + alias Explorer.Chain.Supply.RSK + alias Explorer.Chain.Wei + + @coin_address "0x0000000000000000000000000000000001000006" + + defp wei!(value) do + {:ok, wei} = Wei.cast(value) + wei + end + + test "total is 21_000_000" do + assert RSK.total() == 21_000_000 + end + + describe "circulating/0" do + test "with no balance" do + assert RSK.circulating() == wei!(0) + end + + test "with a balance" do + address = insert(:address, hash: @coin_address) + insert(:block, number: 0) + + insert(:fetched_balance, value: 10, address_hash: address.hash, block_number: 0) + + assert RSK.circulating() == wei!(10) + end + end + + defp date(now, shift \\ []) do + now + |> Timex.shift(shift) + |> Timex.to_date() + end + + defp dec(number) do + Decimal.new(number) + end + + describe "supply_for_days/1" do + test "when there is no balance" do + now = Timex.now() + + assert RSK.supply_for_days(2) == + {:ok, + %{ + date(now, days: -2) => dec(0), + date(now, days: -1) => dec(0), + date(now) => dec(0) + }} + end + + test "when there is a single balance before the days, that balance is used" do + address = insert(:address, hash: @coin_address) + now = Timex.now() + + insert(:block, number: 0, timestamp: Timex.shift(now, days: -10)) + + insert(:fetched_balance, value: 10, address_hash: address.hash, block_number: 0) + + assert RSK.supply_for_days(2) == + {:ok, + %{ + date(now, days: -2) => dec(10), + date(now, days: -1) => dec(10), + date(now) => dec(10) + }} + end + + test "when there is a balance for one of the days, days after it use that balance" do + address = insert(:address, hash: @coin_address) + now = Timex.now() + + insert(:block, number: 0, timestamp: Timex.shift(now, days: -10)) + insert(:block, number: 1, timestamp: Timex.shift(now, days: -1)) + + insert(:fetched_balance, value: 10, address_hash: address.hash, block_number: 0) + + insert(:fetched_balance, value: 20, address_hash: address.hash, block_number: 1) + + assert RSK.supply_for_days(2) == + {:ok, + %{ + date(now, days: -2) => dec(10), + date(now, days: -1) => dec(20), + date(now) => dec(20) + }} + end + + test "when there is a balance for the first day, that balance is used" do + address = insert(:address, hash: @coin_address) + now = Timex.now() + + insert(:block, number: 0, timestamp: Timex.shift(now, days: -10)) + insert(:block, number: 1, timestamp: Timex.shift(now, days: -2)) + insert(:block, number: 2, timestamp: Timex.shift(now, days: -1)) + + insert(:fetched_balance, value: 5, address_hash: address.hash, block_number: 0) + + insert(:fetched_balance, value: 10, address_hash: address.hash, block_number: 1) + + insert(:fetched_balance, value: 20, address_hash: address.hash, block_number: 2) + + assert RSK.supply_for_days(2) == + {:ok, + %{ + date(now, days: -2) => dec(10), + date(now, days: -1) => dec(20), + date(now) => dec(20) + }} + end + + test "when there is a balance for all days, they are each used correctly" do + address = insert(:address, hash: @coin_address) + now = Timex.now() + + insert(:block, number: 0, timestamp: Timex.shift(now, days: -10)) + insert(:block, number: 1, timestamp: Timex.shift(now, days: -2)) + insert(:block, number: 2, timestamp: Timex.shift(now, days: -1)) + insert(:block, number: 3, timestamp: now) + + insert(:fetched_balance, value: 5, address_hash: address.hash, block_number: 0) + insert(:fetched_balance, value: 10, address_hash: address.hash, block_number: 1) + insert(:fetched_balance, value: 20, address_hash: address.hash, block_number: 2) + insert(:fetched_balance, value: 30, address_hash: address.hash, block_number: 3) + + assert RSK.supply_for_days(2) == + {:ok, + %{ + date(now, days: -2) => dec(10), + date(now, days: -1) => dec(20), + date(now) => dec(30) + }} + end + end +end