From c1d8f3792e981af5861c60ee6608fe0e497ff221 Mon Sep 17 00:00:00 2001 From: Artur Troian Date: Mon, 14 Aug 2023 19:10:52 -0400 Subject: [PATCH 1/3] fix: parse price object in price_script_generic.sh --- .../scripts/price_script_generic.sh | 27 +++++++++---------- 1 file changed, 12 insertions(+), 15 deletions(-) diff --git a/charts/akash-provider/scripts/price_script_generic.sh b/charts/akash-provider/scripts/price_script_generic.sh index 55a45f69..6426fda1 100755 --- a/charts/akash-provider/scripts/price_script_generic.sh +++ b/charts/akash-provider/scripts/price_script_generic.sh @@ -69,10 +69,10 @@ function get_akt_price { echo "$usd_per_akt" > $CACHE_FILE fi - # TODO: figure some sort of monitoring to inform the provider in the event API breaks + # TODO: figure some monitoring to inform the provider in the event API breaks fi - # Fail if script can't read CACHE_FILE for some reason + # Fail if the script can't read CACHE_FILE for some reason set -e usd_per_akt=$(cat $CACHE_FILE) echo $usd_per_akt @@ -80,9 +80,9 @@ function get_akt_price { } # If the price parameter is set, new rate calculations will be used -# otherwise, the original rate calculations will be used (for backwards compatibility) +# otherwise, the original rate calculations will be used (for backward compatibility) if ! [[ -z "$price" ]]; then - isDenom=true + hasPrice=true # strip off the .price by setting data_in to .resources data_in=$(echo "$data_in" | jq -r '.resources') @@ -149,15 +149,13 @@ total_cost_uakt="$(printf "%.18f" $rate_per_block_uakt)" # NOTE: max_rate_usd, max_rate_uakt = are per block rates ! -if [[ $isDenom = true ]]; then - denom=$(echo '{"price":"'$price'"}' | jq -r '.price | capture("^[0-9]*\\.?[0-9]*(?.*)") | .denom') - - case "$price" in - - *"uakt") - max_rate_uakt=$(echo '{"price":"'$price'"}' | jq -r '.price | gsub("[^0-9.]"; "")') +if [[ $hasPrice = true ]]; then + denom=$(jq -r '.denom' <<<"$price") + amount=$(jq -r '.amount' <<<"$price") + case "$denom" in + "uakt") # Hint: bc <<< "$a > $b" (if a is greater than b, it will return 1, otherwise 0) - if bc <<< "$rate_per_block_uakt > $max_rate_uakt" | grep -qw 1; then + if bc <<< "$rate_per_block_uakt > $amount" | grep -qw 1; then printf "requested rate is too low. min expected %.18f%s" "$rate_per_block_uakt" "$denom" >&2 exit 1 fi @@ -167,10 +165,9 @@ if [[ $isDenom = true ]]; then ;; # sandbox: Axelar USDC - *"ibc/12C6A0C374171B595A0A9E18B83FA09D295FB1F2D8C6DAA3AC28683471752D84") - max_rate_usd=$(echo '{"price":"'$price'"}' | jq -r '.price | gsub("[^0-9.]"; "")') + "ibc/12C6A0C374171B595A0A9E18B83FA09D295FB1F2D8C6DAA3AC28683471752D84") rate_per_block_usd_normalized=$(bc -l <<<"(${rate_per_block_usd}*1000000)" | awk '{printf "%.18f", $0}') - if bc <<< "$rate_per_block_usd_normalized > $max_rate_usd" | grep -qw 1; then + if bc <<< "$rate_per_block_usd_normalized > $amount" | grep -qw 1; then printf "requested rate is too low. min expected %.18f%s" "$rate_per_block_usd_normalized" "$denom" >&2 exit 1 fi From 3c99a8f9d094cb62f1999cccfb91a19c1ac0d8a5 Mon Sep 17 00:00:00 2001 From: Andrey Arapov Date: Tue, 15 Aug 2023 14:41:38 +0200 Subject: [PATCH 2/3] fix(bid-script): improved price object parsing and validation --- .../scripts/price_script_generic.sh | 37 +++++++++++-------- 1 file changed, 22 insertions(+), 15 deletions(-) diff --git a/charts/akash-provider/scripts/price_script_generic.sh b/charts/akash-provider/scripts/price_script_generic.sh index 6426fda1..a0c4a87a 100755 --- a/charts/akash-provider/scripts/price_script_generic.sh +++ b/charts/akash-provider/scripts/price_script_generic.sh @@ -2,7 +2,7 @@ # WARNING: the runtime of this script should NOT exceed 5 seconds! (Perhaps can be amended via AKASH_BID_PRICE_SCRIPT_PROCESS_TIMEOUT env variable) # Requirements: # curl jq bc mawk ca-certificates -# Version: Aug-14-2023 +# Version: Aug-16-2023 set -o pipefail # Example: @@ -28,16 +28,6 @@ if ! [[ -z $WHITELIST_URL ]]; then fi fi -data_in=$(jq .) -# Pull the pricing data from the deployment request -price=$(echo "$data_in" | jq -r '.price? // empty') - -## DEBUG -if ! [[ -z $DEBUG_BID_SCRIPT ]]; then - echo "$(TZ=UTC date -R)" >> /tmp/${AKASH_OWNER}.log - echo "$data_in" >> /tmp/${AKASH_OWNER}.log -fi - function get_akt_price { # cache AKT price for 60 minutes to reduce the API pressure as well as to slightly accelerate the bidding (+5s) CACHE_FILE=/tmp/aktprice.cache @@ -79,10 +69,29 @@ function get_akt_price { set +e } + +# bid script starts reading the deployment order request specs here (passed by the Akash Provider) +data_in=$(jq .) + +## DEBUG +if ! [[ -z $DEBUG_BID_SCRIPT ]]; then + echo "$(TZ=UTC date -R)" >> /tmp/${AKASH_OWNER}.log + echo "$data_in" >> /tmp/${AKASH_OWNER}.log +fi + +# Pull the pricing data from the deployment request +hasPrice=$(echo "$data_in" | jq -r 'has("price")?') + # If the price parameter is set, new rate calculations will be used # otherwise, the original rate calculations will be used (for backward compatibility) -if ! [[ -z "$price" ]]; then - hasPrice=true +if [[ "$hasPrice" == true ]]; then + isObject=$(jq -r 'if .price?|type == "object" then true else false end' <<<"$data_in") + if [[ "$isObject" != true ]]; then + echo "price must be an object! make sure you are using the latest akash-provider." >&2 + exit 1 + fi + denom=$(jq -r '.price.denom' <<<"$data_in") + amount=$(jq -r '.price.amount' <<<"$data_in") # strip off the .price by setting data_in to .resources data_in=$(echo "$data_in" | jq -r '.resources') @@ -150,8 +159,6 @@ total_cost_uakt="$(printf "%.18f" $rate_per_block_uakt)" # NOTE: max_rate_usd, max_rate_uakt = are per block rates ! if [[ $hasPrice = true ]]; then - denom=$(jq -r '.denom' <<<"$price") - amount=$(jq -r '.amount' <<<"$price") case "$denom" in "uakt") # Hint: bc <<< "$a > $b" (if a is greater than b, it will return 1, otherwise 0) From bdb38de2a6c9c331c7c0f52c5d1e2ec5f8027de2 Mon Sep 17 00:00:00 2001 From: Andrey Arapov Date: Wed, 16 Aug 2023 16:36:03 +0200 Subject: [PATCH 3/3] fix(bid-script): use price_precision --- .../scripts/price_script_generic.sh | 26 ++++++++++--------- 1 file changed, 14 insertions(+), 12 deletions(-) diff --git a/charts/akash-provider/scripts/price_script_generic.sh b/charts/akash-provider/scripts/price_script_generic.sh index a0c4a87a..668c9982 100755 --- a/charts/akash-provider/scripts/price_script_generic.sh +++ b/charts/akash-provider/scripts/price_script_generic.sh @@ -23,7 +23,7 @@ if ! [[ -z $WHITELIST_URL ]]; then fi if ! grep -qw "$AKASH_OWNER" $WHITELIST; then - echo "$AKASH_OWNER is not whitelisted" >&2 + echo -n "$AKASH_OWNER is not whitelisted" >&2 exit 1 fi fi @@ -45,14 +45,14 @@ function get_akt_price { # check price is an integer/floating number re='^[0-9]+([.][0-9]+)?$' if ! [[ $usd_per_akt =~ $re ]]; then - echo "$usd_per_akt is not an integer/floating number!" >&2 + echo -n "$usd_per_akt is not an integer/floating number!" >&2 exit 1 fi # make sure price is in the permitted range if ! (( $(echo "$usd_per_akt > 0" | bc -l) && \ $(echo "$usd_per_akt <= 1000000" | bc -l) )); then - echo "$usd_per_akt is outside the permitted range (>0, <=1000000)" >&2 + echo -n "$usd_per_akt is outside the permitted range (>0, <=1000000)" >&2 exit 1 fi @@ -82,12 +82,15 @@ fi # Pull the pricing data from the deployment request hasPrice=$(echo "$data_in" | jq -r 'has("price")?') +# default price precision to 6 (for backward compatibility) +precision=$(jq -r '.price_precision? // 6' <<<"$data_in") + # If the price parameter is set, new rate calculations will be used # otherwise, the original rate calculations will be used (for backward compatibility) if [[ "$hasPrice" == true ]]; then isObject=$(jq -r 'if .price?|type == "object" then true else false end' <<<"$data_in") if [[ "$isObject" != true ]]; then - echo "price must be an object! make sure you are using the latest akash-provider." >&2 + echo -n "price must be an object! make sure you are using the latest akash-provider." >&2 exit 1 fi denom=$(jq -r '.price.denom' <<<"$data_in") @@ -154,7 +157,7 @@ total_cost_akt_target=$(bc -l <<<"(${total_cost_usd_target}/$usd_per_akt)") total_cost_uakt_target=$(bc -l <<<"(${total_cost_akt_target}*1000000)") rate_per_block_uakt=$(bc -l <<<"(${total_cost_uakt_target}/${blocks_a_month})") rate_per_block_usd=$(bc -l <<<"(${total_cost_usd_target}/${blocks_a_month})") -total_cost_uakt="$(printf "%.18f" $rate_per_block_uakt)" +total_cost_uakt="$(printf "%.*f" $precision $rate_per_block_uakt)" # NOTE: max_rate_usd, max_rate_uakt = are per block rates ! @@ -163,24 +166,24 @@ if [[ $hasPrice = true ]]; then "uakt") # Hint: bc <<< "$a > $b" (if a is greater than b, it will return 1, otherwise 0) if bc <<< "$rate_per_block_uakt > $amount" | grep -qw 1; then - printf "requested rate is too low. min expected %.18f%s" "$rate_per_block_uakt" "$denom" >&2 + printf "requested rate is too low. min expected %.*f%s" "$precision" "$rate_per_block_uakt" "$denom" >&2 exit 1 fi # tell the provider uakt/block rate - echo $total_cost_uakt + printf "%.*f" "$precision" "$total_cost_uakt" ;; # sandbox: Axelar USDC "ibc/12C6A0C374171B595A0A9E18B83FA09D295FB1F2D8C6DAA3AC28683471752D84") - rate_per_block_usd_normalized=$(bc -l <<<"(${rate_per_block_usd}*1000000)" | awk '{printf "%.18f", $0}') + rate_per_block_usd_normalized=$(bc -l <<<"(${rate_per_block_usd}*1000000)" | awk -v precision="$precision" '{printf "%.*f", precision, $0}') if bc <<< "$rate_per_block_usd_normalized > $amount" | grep -qw 1; then - printf "requested rate is too low. min expected %.18f%s" "$rate_per_block_usd_normalized" "$denom" >&2 + printf "requested rate is too low. min expected %.*f%s" "$precision" "$rate_per_block_usd_normalized" "$denom" >&2 exit 1 fi # tell the provider usd/block rate - echo $rate_per_block_usd_normalized + printf "%.*f" "$precision" "$rate_per_block_usd_normalized" ;; *) @@ -192,6 +195,5 @@ if [[ $hasPrice = true ]]; then else # provider only accepts rate in uakt/block when no price is received in structure (backwards compatibility) - echo $total_cost_uakt + printf "%.f" "$total_cost_uakt" fi -