#!/usr/bin/env bash ### ### arching-kaos-tools ### Tools to interact and build an Arching Kaos Infochain ### Copyright (C) 2021 - 2025 kaotisk ### ### This program is free software: you can redistribute it and/or modify ### it under the terms of the GNU General Public License as published by ### the Free Software Foundation, either version 3 of the License, or ### (at your option) any later version. ### ### This program is distributed in the hope that it will be useful, ### but WITHOUT ANY WARRANTY; without even the implied warranty of ### MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ### GNU General Public License for more details. ### ### You should have received a copy of the GNU General Public License ### along with this program. If not, see . ### source $AK_LIBDIR/_ak_log source $AK_LIBDIR/_ak_ipfs source $AK_LIBDIR/_ak_zblock source $AK_LIBDIR/_ak_script function _ak_zchain_reset(){ _ak_log_info "Reseting AK_ZLATEST to AK_ZGENESIS" cp $AK_ZGENESIS $AK_ZLATEST if [ $? -ne 0 ] then _ak_log_error "Failed to reset AK_ZLATEST to AK_ZGENESIS" exit 1 fi _ak_log_info "Checking if /zarchive directory exists" _ak_ipfs_files_stat /zarchive > /dev/null if [ $? -ne 0 ] then _ak_ipfs_files_mkdir /zarchive if [ $? -ne 0 ] then _ak_log_error "Could not create directory /zarchive" exit 1 fi else _ak_log_info "/zarchive directory exists" fi _ak_log_info "Checking if /zlatest file exists" _ak_ipfs_files_stat /zlatest > /dev/null 2>&1 if [ $? -eq 0 ] then _ak_log_info "Archive the previous AK_ZLATEST" _ak_ipfs_files_cp /zlatest /zarchive/$(date -u +%s)-$(_ak_ipfs_files_stat /zlatest | head -n 1) if [ $? -ne 0 ] then _ak_log_error "Failed to copy /zlatest to /zarchive" exit 1 fi _ak_log_info "Removing previous /zlatest entry" _ak_ipfs_files_rm /zlatest if [ $? -ne 0 ] then _ak_log_error "Failed to remove /zlatest" exit 1 fi else _ak_log_info "/zlatest not found, skipping backup" fi _ak_log_info "Copying reset AK_ZLATEST" CZLATEST="$(cat $AK_ZLATEST)" _ak_ipfs_files_cp /ipfs/$CZLATEST /zlatest if [ $? -ne 0 ] then _ak_log_error "Failed to copy AK_ZLATEST to /zlatest" exit 1 fi _ak_log_info "Publishing new (reset) AK_ZLATEST" _ak_ipfs_name_publish --key=zchain /ipfs/$(cat $AK_ZLATEST) if [ $? -ne 0 ] then _ak_log_error "Failed to publish updated zchain" exit 1 fi _ak_config_publish if [ $? -ne 0 ] then _ak_log_error "Could not publish new configuration" exit 1 fi _ak_log_info "Reset was successful" exit 0 } function _ak_zchain_rebase(){ if [ ! -n "$1" ]; then exit 1; fi ZTARGET="$1" echo "Reseting ZLATEST to ZTARGET" echo $ZTARGET > $AK_ZLATEST if [ $? != 0 ]; then exit 1; fi echo "Make sure /zarchive folder exists within IPFS FS" _ak_ipfs_files_mkdir /zarchive if [ $? != 0 ]; then echo "Folder already there"; fi echo "Archive the previous ZLATEST" _ak_ipfs_files_cp /zlatest /zarchive/$(date -u +%s)-$(_ak_ipfs_files_stat /zlatest | head -n 1) if [ $? != 0 ]; then exit 1; fi echo "Removing previous /zlatest entry" _ak_ipfs_files_rm /zlatest if [ $? != 0 ]; then exit 1; fi echo "Copying rebased ZLATEST" CZLATEST="$(cat $AK_ZLATEST)" _ak_ipfs_files_cp /ipfs/$CZLATEST /zlatest if [ $? != 0 ]; then exit 1; fi echo "Publishing new (rebased) ZLATEST" _ak_ipfs_name_publish --key=zchain /ipfs/$(cat $AK_ZLATEST) if [ $? != 0 ]; then exit 1; fi _ak_config_publish if [ $? -ne 0 ] then _ak_log_error "Could not publish new configuration" exit 1 fi echo "Rebase was successful" exit 0 } function _ak_zchain_extract_cids(){ if [ ! -z $1 ] && [ -n "$1" ] then _ak_zchain_crawl $1 | jq -M | grep Qm | sed -e 's/".*"://g; s/ //g; s/[{,"]//g' | sort | uniq else _ak_zchain_crawl | jq -M | grep Qm | sed -e 's/".*"://g; s/ //g; s/[{,"]//g' | sort | uniq fi } function _ak_zchain_extract_cids_limit(){ if [ ! -z $1 ] && [ -n "$1" ] && [ ! -z $2 ] && [ -n "$2" ] then _ak_zchain_crawl -l $2 $1 | jq -M | grep Qm | sed -e 's/".*"://g; s/ //g; s/[{,"]//g' | sort | uniq else _ak_zchain_crawl | jq -M | grep Qm | sed -e 's/".*"://g; s/ //g; s/[{,"]//g' | sort | uniq fi } function _ak_zchain_extract_data_cids(){ if [ ! -z $1 ] then _ak_zchain_crawl $1 | jq | grep ipfs | awk '{print $2}' | sed -e 's/"//g;s/,//g' else _ak_zchain_crawl | jq | grep ipfs | awk '{print $2}' | sed -e 's/"//g;s/,//g' fi } function _ak_zchain_calculate_size(){ temp="$(_ak_make_temp_directory)" cd $temp if [ ! -z $1 ] && [ -n "$1" ] then _ak_zchain_extract_cids $1 > to_stats else _ak_zchain_extract_cids > to_stats fi sum=0 ; while IFS="" read -r p || [ -n "$p" ] do if [ "$p" != "" ] then _ak_ipfs_get $p num="$( (du -b $p || du -A $p)2>/dev/null | cut -d $'\t' -f 1)" else num=0 fi sum=$(expr $sum + $num ) done < to_stats echo "Chain is : $sum bytes" cd ~ rm -rf $temp } function _ak_zchain_crawl(){ entrance="$(cat $AK_ZLATEST)" verify=1 limit=0 fromIpns=0 while [ "$#" ]; do case "$1" in -h | --help) printf "Zchain crawler ============== ak zchain --crawl [-N | --no-verify] [-l | --limit ] [zblock] ak zchain --crawl [-N | --no-verify] [-l | --limit ] -n Usage: --help, -h Print this help and exit --chain , -n Crawl specified chain --no-verify, -N Don't verify signatures Specify IPFS CID for entrance Note that combined flags don't work for now Running with no flags crawls your chain based on AK_ZLATEST environment variable " exit 1 ;; -l | --limit) limit=$2 shift 2 ;; -N | --no-verify) verify=0 shift ;; -n | --chain | --ipns) fromIpns=1 ipns=$1 shift ol=$1 entrance="$(_ak_ipns_resolve $1)" if [ $? -ne 0 ] then _ak_log_error "Could not resolve IPNS name" exit 1 fi shift ;; *) test="$1" if [ ! -z "$test" ] && [ $fromIpns -eq 0 ] then _ak_ipfs_cid_v0_check "$test" entrance="$test" elif [ -z "$entrance" ] && [ $fromIpns -eq 1 ] then entrance="$(cat $AK_ZLATEST)" fi break esac done # We assign the IPFS CIDv0 of an empty file as this is used # as our GENESIS block, hence the "seed" that the tree grows # from. seed="$(cat $AK_ZGENESIS)" # We assume that we found the entrance inside a block, hence # ZBLOCK is labeled as previous zblock="$entrance" # Enter temp folder TEMPASSIN="$(_ak_make_temp_directory)" cd $TEMPASSIN counter=0 # The loop # We break the loop from inside the loop while true do if [ $counter -eq 0 ] then echo -n '[' fi counter=$(($counter + 1)) _ak_zblock_show "$zblock" if [ $limit -ne 0 ] && [ $limit -eq $counter ] then echo -n ']' exit 0 else echo -n ',' fi done } function _ak_zchain_crawl_self(){ _ak_zchain_crawl } function _ak_zchain_crawl_remote_ipfs(){ _ak_zchain_crawl $1 } function _ak_zchain_crawl_remote_ipns(){ _ak_zchain_crawl -n $1 } function _ak_zchain_get_latest(){ _ak_ipfs_files_stat /zlatest | head -n 1| tr -d '\n' } # _ak_zchain_announce(){} # PROGRAM="$(basename $0)" # printf '[%s]\n' "$PROGRAM" # printf "TEST\t/v0/announce/zchain\n" # # curl http://127.0.0.1:8610/v0/announce/zchain # printf "\t01:\tendpoint with valid data" # curl \ # --connect-timeout 3 \ # -POST http://localhost:8610/v0/announce/zchain \ # --header 'Content-Type: application/json' \ # --data-raw '{"zchain":"k51qzi5uqu5dgapvk7bhxmchuqya9immqdpbz0f1r91ckpdqzub63afn3d5apr"}' 2>/dev/null | jq -M -c > /dev/null # if [ $? -eq 0 ] # then # printf "\t\t\033[0;32mPASSED\033[0;0m" # else # printf "\t\033[0;31mFAILED\033[0;0m" # fi # printf "\n" # # printf "\t02:\tendpoint with invalid data" # curl \ # --connect-timeout 3 \ # -POST http://localhost:8610/v0/announce/zchain \ # --header 'Content-Type: application/json' \ # --data-raw '{"zchain":"k51qzi5uqu5dgapvk7bhxmchuqya9immqdpbz0f1r91ckpdqzub63afn3d5aar"}' 2>/dev/null | jq -M -c > /dev/null # if [ $? -eq 0 ] # then # printf "\t\t\033[0;32mPASSED\033[0;0m" # else # printf "\t\033[0;31mFAILED\033[0;0m" # fi # printf "\n" # # printf "\t03:\tendpoint no data" # curl \ # --connect-timeout 3 \ # -POST http://localhost:8610/v0/announce/zchain \ # --header 'Content-Type: application/json' \ # --data-raw '{"zchain":""}' 2>/dev/null | jq -M -c > /dev/null # if [ $? -eq 0 ] # then # printf "\t\t\t\033[0;32mPASSED\033[0;0m" # else # printf "\t\t\t\033[0;31mFAILED\033[0;0m" # fi # printf "\n" # } # _ak_zchain_chk(){ # ## # ## -h, --help Prints this help message # ## # fullprogrampath="$(realpath $0)" # PROGRAM=$(basename $0) # descriptionString="Quick description" # source $AK_LIBDIR/_ak_log # source $AK_LIBDIR/_ak_script # source $AK_LIBDIR/_ak_ipfs # # fix="0" # usage(){ # echo "zchain-chk - Check and fix zchain" # echo "---------------------------------" # echo "Usage:" # echo " --help, -h Print this help and exit" # echo " --chain , -n Crawl specified chain" # echo " --fix #TODO Fix your chain" # echo "" # echo "Note that combined flags don't work for now" # echo "Running with no flags crawls your chain" # } # # if [ ! -z "$1" ] && [ "$1" == "-h" ] || [ "$1" == "--help" ] # then # usage # exit # elif [ ! -z "$1" ] && [ "$1" == "-f" ] || [ "$1" == "--fix" ] # then # fix="1" # entrance="$(cat $AK_ZLATEST)" # elif [ ! -z "$1" ] && [ "$1" == "-n" ] || [ "$1" == "--chain" ] # then # entrance="$(_ak_ipns_resolve $2)" # elif [ ! -z "$1" ] # then # entrance="$1" # else # # By default we ak-enter from the latest block # # We can alter this by changing this value # entrance="$(cat $AK_ZLATEST)" # fi # # # We assign the IPFS CIDv0 of an empty file as this is used # # as our GENESIS block, hence the "seed" that the tree grows # # from. # seed="$(cat $AK_ZGENESIS)" # # # We assume that we found the entrance inside a block, hence # # ZBLOCK is labeled as previous # zblock="$entrance" # declare -A blocks_found # # # Enter temp folder # TEMPASSIN="$(_ak_make_temp_directory)" # cd $TEMPASSIN # counter=0 # # # The loop # # We break the loop from inside the loop # while true # do # if [ $counter == 0 ] # then # echo 'Start checking' # fi # counter=$(expr $counter + 1) # # # Check if $zblock exists as variable # if [ ! -v $zblock ] # then # # Check if it is not our seed cause if it is we skip this part # if [ "$zblock" != "$seed" ] # then # # Reset timestamp since it's introduced later # timestamp='' # # Announce to stdout which ZBLOCK is being read at the moment # _ak_log_info "Examining $zblock" # # # We create files named after each ZBLOCK IPFS CID for later # # reference. Files are empty. # touch $AK_ZBLOCKDIR/$zblock # _ak_log_info "Created reference" # # # We concatenate the zblock's contents, pipe them through filter # # ak-json2bash and output them to tmp-file # _ak_ipfs_cat $zblock | ak-json2bash > tmp-zblock # _ak_log_info "ZBLOCK $zblock READ" # # # Supposingly you are on a safe environment and you only have # # access to your chain, I would consider mild secure to source # # the files into your bash. # # File an issue/pull request if you think it can be done better!! # source tmp-zblock # _ak_log_info "ZBLOCK SOURCED" # # # Same as above applies to BLOCK and DATA subparts of each ZBLOCK # # BLOCKS # _ak_ipfs_cat $block | ak-json2bash > tmp-block # source tmp-block # _ak_log_info "BLOCK $block SOURCED" # touch $AK_BLOCKDIR/$block # _ak_log_info "BLOCK REFERENCED" # module="$(echo $action | sed -e 's/\// /g' | awk '{ print $1 }')" # _ak_log_info "DATA is $module module." # command="$(echo $action | sed -e 's/\// /g' | awk '{ print $2 }')" # _ak_log_info "COMMAND is $command" # if [ ! -v $timestamp ] # then # echo "$timestamp : $zblock -> $block -> $previous" # blocks_found[$counter]="$block" # fi # touch $AK_DATADIR/$data # # # Now, since we sourced the BLOCK to our terminal, we can search # # for $previous variable. In case we don't find one, we exit with # # code 0 # if [ -v $previous ] # then # _ak_log_error "Block $block has no previous zblock" # echo "Chain with no genesis" # if [ "$fix" == "1" ] # then # echo "LOL" # else # echo "Blocks found and need repacking..." # for value in ${blocks_found[@]} # do # echo $value # _ak_ipfs_cat $value | jq -M # done # fi # exit 0 # # Otherwise, we inform of the sequence # else # zblock=$previous # fi # # Now check if it is equal to the seed # # which apparently means we reached the seed. # elif [ "$zblock" == "$seed" ] # then # echo "Chain is OK with GENESIS block = $seed" # _ak_log_info "Counter $counter" # exit 0 # fi # # And finally, if nothing is there exit with error # else # _ak_log_error "Check not passed... No previous IPFS CID" # exit 1 # fi # done # }