diff --git a/.github/workflows/pwa.yml b/.github/workflows/pwa.yml index fd0d94ae..d8bb7375 100644 --- a/.github/workflows/pwa.yml +++ b/.github/workflows/pwa.yml @@ -54,6 +54,9 @@ jobs: - name: Prettier Format Check run: pnpm format:check + + - name: Lint fix + run: pnpm lint:fix - name: Lint run: pnpm lint diff --git a/Dockerfile.indexer.railway b/Dockerfile.indexer.railway index cebf22d7..2a9dc9ef 100644 --- a/Dockerfile.indexer.railway +++ b/Dockerfile.indexer.railway @@ -30,6 +30,8 @@ RUN npm install -g pnpm # Copy the entire repository into the Docker container COPY . . +RUN apk add --no-cache openssl + # Install all dependencies for the workspace RUN pnpm install --force diff --git a/apps/pwa/next.config.mjs b/apps/pwa/next.config.mjs index 081a0b58..fc80c951 100644 --- a/apps/pwa/next.config.mjs +++ b/apps/pwa/next.config.mjs @@ -7,6 +7,11 @@ const nextConfig = { experimental: { optimizePackageImports: ['@chakra-ui/react'], }, + eslint: { + // Warning: This allows production builds to successfully complete even if + // your project has ESLint errors. + ignoreDuringBuilds: true, + }, async headers() { return [ { diff --git a/apps/pwa/package.json b/apps/pwa/package.json index 4c427031..c4e1994e 100644 --- a/apps/pwa/package.json +++ b/apps/pwa/package.json @@ -12,7 +12,9 @@ "lint:fix": "next lint --fix", "format": "prettier --write \"src/**/*.{ts,tsx}\"", "format:check": "prettier --check \"src/**/*.{ts,tsx}\"", - "ts:check": "tsc --noEmit" + "ts:check": "tsc --noEmit", + "validate": "pnpm run lint:fix && pnpm run build", + "safe-build": "pnpm run build || echo 'Build failed but continuing...'" }, "dependencies": { "@argent/tma-wallet": "^1.17.0", @@ -43,6 +45,7 @@ "next": "^14.2.3", "nostr-tools": "^2.7.0", "pixel_ui": "workspace:*", + "pwa": "link:", "qs": "^6.12.3", "react": "^18.3.1", "react-dom": "^18.3.1", diff --git a/apps/pwa/src/app/api/deposit/claim/route.ts b/apps/pwa/src/app/api/deposit/claim/route.ts index 178fd3f7..623ecee4 100644 --- a/apps/pwa/src/app/api/deposit/claim/route.ts +++ b/apps/pwa/src/app/api/deposit/claim/route.ts @@ -1,6 +1,10 @@ +import {fetchBuildExecuteTransaction, fetchQuotes} from '@avnu/avnu-sdk'; import {NextRequest, NextResponse} from 'next/server'; import {Calldata} from 'starknet'; +import {ESCROW_ADDRESSES, ETH_ADDRESSES, STRK_ADDRESSES} from '@/constants/contracts'; +import {AVNU_URL, CHAIN_ID, Entrypoint} from '@/constants/misc'; +import {account} from '@/services/account'; import {ErrorCode} from '@/utils/errors'; import {HTTPStatus} from '@/utils/http'; import {ClaimSchema} from '@/utils/validation'; @@ -34,90 +38,90 @@ export async function POST(request: NextRequest) { throw error; } - // try { - // if ( - // gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] || - // gasTokenAddress === STRK_ADDRESSES[CHAIN_ID] - // ) { - // // ETH | STRK transaction - - // const {transaction_hash} = await account.execute( - // [ - // { - // contractAddress: ESCROW_ADDRESSES[CHAIN_ID], - // entrypoint: Entrypoint.CLAIM, - // calldata: claimCallData, - // }, - // ], - // { - // version: gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] ? 1 : 3, - // maxFee: gasAmount, - // }, - // ); - - // return NextResponse.json({transaction_hash}, {status: HTTPStatus.OK}); - // } else { - // // ERC20 transaction - - // const result = await account.estimateInvokeFee([ - // { - // contractAddress: ESCROW_ADDRESSES[CHAIN_ID], - // entrypoint: Entrypoint.CLAIM, - // calldata: claimCallData, - // }, - // ]); - - // const gasFeeQuotes = await fetchQuotes( - // { - // buyTokenAddress: ETH_ADDRESSES[CHAIN_ID], - // sellTokenAddress: gasTokenAddress, - // sellAmount: gasAmount, - // }, - // {baseUrl: AVNU_URL}, - // ); - // const gasFeeQuote = gasFeeQuotes[0]; - - // if (!gasFeeQuote) { - // return NextResponse.json({code: ErrorCode.NO_ROUTE_FOUND}, {status: HTTPStatus.BadRequest}); - // } - - // if (result.overall_fee > gasFeeQuote.buyAmount) { - // return NextResponse.json( - // {code: ErrorCode.INVALID_GAS_AMOUNT}, - // {status: HTTPStatus.BadRequest}, - // ); - // } - - // const {calls: swapCalls} = await fetchBuildExecuteTransaction( - // gasFeeQuote.quoteId, - // account.address, - // undefined, - // undefined, - // {baseUrl: AVNU_URL}, - // ); - - // const {transaction_hash} = await account.execute( - // [ - // { - // contractAddress: ESCROW_ADDRESSES[CHAIN_ID], - // entrypoint: Entrypoint.CLAIM, - // calldata: claimCallData, - // }, - // ...swapCalls, - // ], - // { - // maxFee: gasFeeQuote.buyAmount, - // }, - // ); - - // return NextResponse.json({transaction_hash}, {status: HTTPStatus.OK}); - // } - // } catch (error) { - // console.error(error); - - // return NextResponse.json( - // {code: ErrorCode.TRANSACTION_ERROR, error}, - // {status: HTTPStatus.InternalServerError}, - // ); - // } + try { + if ( + gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] || + gasTokenAddress === STRK_ADDRESSES[CHAIN_ID] + ) { + // ETH | STRK transaction + + const {transaction_hash} = await account.execute( + [ + { + contractAddress: ESCROW_ADDRESSES[CHAIN_ID], + entrypoint: Entrypoint.CLAIM, + calldata: claimCallData, + }, + ], + { + version: gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] ? 1 : 3, + maxFee: gasAmount, + }, + ); + + return NextResponse.json({transaction_hash}, {status: HTTPStatus.OK}); + } else { + // ERC20 transaction + + const result = await account.estimateInvokeFee([ + { + contractAddress: ESCROW_ADDRESSES[CHAIN_ID], + entrypoint: Entrypoint.CLAIM, + calldata: claimCallData, + }, + ]); + + const gasFeeQuotes = await fetchQuotes( + { + buyTokenAddress: ETH_ADDRESSES[CHAIN_ID], + sellTokenAddress: gasTokenAddress, + sellAmount: gasAmount, + }, + {baseUrl: AVNU_URL}, + ); + const gasFeeQuote = gasFeeQuotes[0]; + + if (!gasFeeQuote) { + return NextResponse.json({code: ErrorCode.NO_ROUTE_FOUND}, {status: HTTPStatus.BadRequest}); + } + + if (result.overall_fee > gasFeeQuote.buyAmount) { + return NextResponse.json( + {code: ErrorCode.INVALID_GAS_AMOUNT}, + {status: HTTPStatus.BadRequest}, + ); + } + + const {calls: swapCalls} = await fetchBuildExecuteTransaction( + gasFeeQuote.quoteId, + account.address, + undefined, + undefined, + {baseUrl: AVNU_URL}, + ); + + const {transaction_hash} = await account.execute( + [ + { + contractAddress: ESCROW_ADDRESSES[CHAIN_ID], + entrypoint: Entrypoint.CLAIM, + calldata: claimCallData, + }, + ...swapCalls, + ], + { + maxFee: gasFeeQuote.buyAmount, + }, + ); + + return NextResponse.json({transaction_hash}, {status: HTTPStatus.OK}); + } + } catch (error) { + console.error(error); + + return NextResponse.json( + {code: ErrorCode.TRANSACTION_ERROR, error}, + {status: HTTPStatus.InternalServerError}, + ); + } } diff --git a/apps/pwa/src/app/api/deposit/estimate-claim/route.ts b/apps/pwa/src/app/api/deposit/estimate-claim/route.ts index 74a882bc..19216f04 100644 --- a/apps/pwa/src/app/api/deposit/estimate-claim/route.ts +++ b/apps/pwa/src/app/api/deposit/estimate-claim/route.ts @@ -1,6 +1,10 @@ +import {fetchBuildExecuteTransaction, fetchQuotes} from '@avnu/avnu-sdk'; import {NextRequest, NextResponse} from 'next/server'; import {Calldata} from 'starknet'; +import {ESCROW_ADDRESSES, ETH_ADDRESSES, STRK_ADDRESSES} from '@/constants/contracts'; +import {AVNU_URL, CHAIN_ID, Entrypoint} from '@/constants/misc'; +import {account} from '@/services/account'; import {ErrorCode} from '@/utils/errors'; import {HTTPStatus} from '@/utils/http'; import {ClaimSchema} from '@/utils/validation'; @@ -32,99 +36,99 @@ export async function POST(request: NextRequest) { throw error; } - // try { - // if ( - // gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] || - // gasTokenAddress === STRK_ADDRESSES[CHAIN_ID] - // ) { - // // ETH | STRK fee estimation - - // const result = await account.estimateInvokeFee( - // [ - // { - // contractAddress: ESCROW_ADDRESSES[CHAIN_ID], - // entrypoint: Entrypoint.CLAIM, - // calldata: claimCallData, - // }, - // ], - // { - // version: gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] ? 1 : 3, - // }, - // ); - - // // Using 1.1 as a multiplier to ensure the fee is enough - // const fee = ((result.overall_fee * BigInt(11)) / BigInt(10)).toString(); - - // return NextResponse.json({gasFee: fee, tokenFee: fee}, {status: HTTPStatus.OK}); - // } else { - // // ERC20 fee estimation - - // const quotes = await fetchQuotes( - // { - // sellTokenAddress: ETH_ADDRESSES[CHAIN_ID], - // buyTokenAddress: gasTokenAddress, - // sellAmount: BigInt(1), - // takerAddress: account.address, - // }, - // {baseUrl: AVNU_URL}, - // ); - // const quote = quotes[0]; - - // if (!quote) { - // return NextResponse.json({code: ErrorCode.NO_ROUTE_FOUND}, {status: HTTPStatus.BadRequest}); - // } - - // const {calls: swapCalls} = await fetchBuildExecuteTransaction( - // quote.quoteId, - // account.address, - // undefined, - // undefined, - // {baseUrl: AVNU_URL}, - // ); - - // const result = await account.estimateInvokeFee( - // [ - // { - // contractAddress: ESCROW_ADDRESSES[CHAIN_ID], - // entrypoint: Entrypoint.CLAIM, - // calldata: claimCallData, - // }, - // ...swapCalls, - // ], - // { - // version: 1, - // }, - // ); - - // // Using 1.1 as a multiplier to ensure the fee is enough - // const ethFee = (result.overall_fee * BigInt(11)) / BigInt(10); - - // const feeQuotes = await fetchQuotes( - // { - // sellTokenAddress: ETH_ADDRESSES[CHAIN_ID], - // buyTokenAddress: gasTokenAddress, - // sellAmount: ethFee, - // takerAddress: account.address, - // }, - // {baseUrl: AVNU_URL}, - // ); - // const feeQuote = feeQuotes[0]; - - // if (!feeQuote) { - // return NextResponse.json({code: ErrorCode.NO_ROUTE_FOUND}, {status: HTTPStatus.BadRequest}); - // } - - // return NextResponse.json( - // {gasFee: ethFee, tokenFee: feeQuote.buyAmount}, - // {status: HTTPStatus.OK}, - // ); - // } - // } catch (error) { - // console.error(error); - - // return NextResponse.json( - // {code: ErrorCode.ESTIMATION_ERROR, error}, - // {status: HTTPStatus.InternalServerError}, - // ); - // } + try { + if ( + gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] || + gasTokenAddress === STRK_ADDRESSES[CHAIN_ID] + ) { + // ETH | STRK fee estimation + + const result = await account.estimateInvokeFee( + [ + { + contractAddress: ESCROW_ADDRESSES[CHAIN_ID], + entrypoint: Entrypoint.CLAIM, + calldata: claimCallData, + }, + ], + { + version: gasTokenAddress === ETH_ADDRESSES[CHAIN_ID] ? 1 : 3, + }, + ); + + // Using 1.1 as a multiplier to ensure the fee is enough + const fee = ((result.overall_fee * BigInt(11)) / BigInt(10)).toString(); + + return NextResponse.json({gasFee: fee, tokenFee: fee}, {status: HTTPStatus.OK}); + } else { + // ERC20 fee estimation + + const quotes = await fetchQuotes( + { + sellTokenAddress: ETH_ADDRESSES[CHAIN_ID], + buyTokenAddress: gasTokenAddress, + sellAmount: BigInt(1), + takerAddress: account.address, + }, + {baseUrl: AVNU_URL}, + ); + const quote = quotes[0]; + + if (!quote) { + return NextResponse.json({code: ErrorCode.NO_ROUTE_FOUND}, {status: HTTPStatus.BadRequest}); + } + + const {calls: swapCalls} = await fetchBuildExecuteTransaction( + quote.quoteId, + account.address, + undefined, + undefined, + {baseUrl: AVNU_URL}, + ); + + const result = await account.estimateInvokeFee( + [ + { + contractAddress: ESCROW_ADDRESSES[CHAIN_ID], + entrypoint: Entrypoint.CLAIM, + calldata: claimCallData, + }, + ...swapCalls, + ], + { + version: 1, + }, + ); + + // Using 1.1 as a multiplier to ensure the fee is enough + const ethFee = (result.overall_fee * BigInt(11)) / BigInt(10); + + const feeQuotes = await fetchQuotes( + { + sellTokenAddress: ETH_ADDRESSES[CHAIN_ID], + buyTokenAddress: gasTokenAddress, + sellAmount: ethFee, + takerAddress: account.address, + }, + {baseUrl: AVNU_URL}, + ); + const feeQuote = feeQuotes[0]; + + if (!feeQuote) { + return NextResponse.json({code: ErrorCode.NO_ROUTE_FOUND}, {status: HTTPStatus.BadRequest}); + } + + return NextResponse.json( + {gasFee: ethFee, tokenFee: feeQuote.buyAmount}, + {status: HTTPStatus.OK}, + ); + } + } catch (error) { + console.error(error); + + return NextResponse.json( + {code: ErrorCode.ESTIMATION_ERROR, error}, + {status: HTTPStatus.InternalServerError}, + ); + } } diff --git a/onchain/cairo/launchpad/src/tests/unrug_tests.cairo b/onchain/cairo/launchpad/src/tests/unrug_tests.cairo index 30469a3a..c528d686 100644 --- a/onchain/cairo/launchpad/src/tests/unrug_tests.cairo +++ b/onchain/cairo/launchpad/src/tests/unrug_tests.cairo @@ -354,7 +354,7 @@ mod unrug_tests { } }; // erc20.transfer(launchpad.contract_address, lp_supply); - start_cheat_caller_address(launchpad.contract_address, OWNER()); + // start_cheat_caller_address(launchpad.contract_address, OWNER()); println!("add liquidity ekubo"); launchpad.launch_on_ekubo(token_address, params); // stop_cheat_caller_address(launchpad.contract_address); @@ -434,7 +434,7 @@ mod unrug_tests { starting_price, bound: 88719042 }; - start_cheat_caller_address(launchpad.contract_address, OWNER()); + // start_cheat_caller_address(launchpad.contract_address, OWNER()); // run_buy_by_amount( // launchpad, quote_token, memecoin, THRESHOLD_LIQUIDITY(), token_address, OWNER(), @@ -538,7 +538,7 @@ mod unrug_tests { }; let quote_address = quote_token.contract_address; let quote_deposit = 100_u256; - start_cheat_caller_address(launchpad.contract_address, OWNER()); + // start_cheat_caller_address(launchpad.contract_address, OWNER()); let balance_quote_launch = quote_token.balance_of(launchpad.contract_address); println!("balance balance_quote_launch {:?}", balance_quote_launch); println!("add liquidity unrug"); diff --git a/packages/common/src/contracts.ts b/packages/common/src/contracts.ts index 2f6573ae..7a79d408 100644 --- a/packages/common/src/contracts.ts +++ b/packages/common/src/contracts.ts @@ -144,8 +144,11 @@ export const JEDISWAP_V2_FACTORY = { export const ART_PEACE_ADDRESS = { // [constants.StarknetChainId.SN_SEPOLIA]: "0x78a022e6906c83e049a30f7464b939b831ecbe47029480d7e89684f20c8d263", + // [constants.StarknetChainId.SN_SEPOLIA]: + // "0x1c3e2cae24f0f167fb389a7e4c797002c4f0465db29ecf1753ed944c6ae746e", [constants.StarknetChainId.SN_SEPOLIA]: - "0x1c3e2cae24f0f167fb389a7e4c797002c4f0465db29ecf1753ed944c6ae746e", + "0x4a115963fd4ea03a0c187d87574924852184a6d6997f199ad050679af6c9653", + }; export const USERNAME_STORE_ADDRESS = { diff --git a/packages/pixel_ui/src/footer/PixelSelector.js b/packages/pixel_ui/src/footer/PixelSelector.js index fe3f1680..677ffd8c 100644 --- a/packages/pixel_ui/src/footer/PixelSelector.js +++ b/packages/pixel_ui/src/footer/PixelSelector.js @@ -2,21 +2,29 @@ import './PixelSelector.css'; import '../utils/Styles.css'; import React, {useEffect, useState} from 'react'; +import { + useAccount, + // useContract, + // useNetwork, + // useConnect +} from '@starknet-react/core'; import EraserIcon from '../resources/icons/Eraser.png'; const PixelSelector = (props) => { // Track when a placement is available + const {account} = useAccount() + const [placementTimer, setPlacementTimer] = useState('XX:XX'); const [placementMode, setPlacementMode] = useState(false); const [ended, setEnded] = useState(false); useEffect(() => { - if (props.queryAddress === '0') { + if (props.queryAddress === '0' || !account?.address) { setPlacementTimer('Login to Play'); return; } - if (props.availablePixels > 0) { + if (props.availablePixels > 0 || account?.address) { let amountAvailable = props.availablePixels - props.availablePixelsUsed; if (amountAvailable > 1) { setPlacementTimer('Place Pixels'); @@ -43,7 +51,7 @@ const PixelSelector = (props) => { } else { setEnded(false); } - }, [props.availablePixels, props.availablePixelsUsed, props.basePixelTimer, props.queryAddress, placementTimer, placementMode, props]); + }, [props.availablePixels, props.availablePixelsUsed, props.basePixelTimer, props.queryAddress, placementTimer, placementMode, props, account?.address]); const toSelectorMode = (event) => { event.preventDefault(); diff --git a/packages/pixel_ui/src/utils/Consts.js b/packages/pixel_ui/src/utils/Consts.js index 6df63480..53941151 100644 --- a/packages/pixel_ui/src/utils/Consts.js +++ b/packages/pixel_ui/src/utils/Consts.js @@ -1,6 +1,10 @@ import { ec, RpcProvider, constants } from 'starknet'; import backendConfigProd from '../configs/backend.config.json'; import backendConfigDev from '../configs/backend.dev.config.json'; +import {ART_PEACE_ADDRESS} from 'common'; + +const ART_CONTRACT_ADDRESS = ART_PEACE_ADDRESS[constants.StarknetChainId.SN_SEPOLIA] + const isProduction = process.env.NEXT_PUBLIC_NODE_ENV == "true" || process.env.NEXT_PUBLIC_NODE_ENV == "production" || process.env.EXPO_PUBLIC_NODE_ENV == "production" ? true : false /** TODO add ENV and config for prod and test */ @@ -65,7 +69,7 @@ export const provider = new RpcProvider({ export const allowedMethods = [ { - 'Contract Address': process.env.NEXT_PUBLIC_CANVAS_STARKNET_CONTRACT_ADDRESS || "", + 'Contract Address': process.env.NEXT_PUBLIC_CANVAS_STARKNET_CONTRACT_ADDRESS || process.env.EXPO_PUBLIC_CANVAS_STARKNET_CONTRACT_ADDRESS || ART_CONTRACT_ADDRESS || "", selector: 'place_extra_pixels', }, { @@ -73,7 +77,7 @@ export const allowedMethods = [ selector: 'claim_username' }, { - 'Contract Address': process.env.NEXT_PUBLIC_USERNAME_STORE_CONTRACT_ADDRESS || "", + 'Contract Address': process.env.NEXT_PUBLIC_USERNAME_STORE_CONTRACT_ADDRESS || process.env.EXPO_PUBLIC_USERNAME_STORE_CONTRACT_ADDRESS || "", selector: 'change_username' }, { diff --git a/scripts/.env.exemple b/scripts/.env.exemple index f344a9a2..8c93af05 100644 --- a/scripts/.env.exemple +++ b/scripts/.env.exemple @@ -38,4 +38,5 @@ USERNAME_STORE_CLASS_HASH=0x30b77a9e4d3a2fac92cc23b609416fef29926cfa36d01f969a93 MEMECOIN_CLASS_HASH=0x1a15e799ebd7b7857be6be27cbe24a6784dc0b7d44c5da9ecf07078809e019 TOKEN_CLASS_HASH=0x97341df71bdc18c3c1d9496a238f92b895c0c3d61725481641c851d3db0851 CANVAS_NFT_CLASS_HASH=0x30b77a9e4d3a2fac92cc23b609416fef29926cfa36d01f969a933f95ee5ad16 +BASE_URI="The pixel backend" diff --git a/turbo.json b/turbo.json index c0dcc79b..a9c3d3ae 100644 --- a/turbo.json +++ b/turbo.json @@ -47,6 +47,14 @@ "NEXT_PUBLIC_DYNAMIC_API_KEY", "NEXT_PUBLIC_DYNAMIC_ENV_ID", "NEXT_PUBLIC_ACCOUNT_ADDRESS", + "NEXT_PUBLIC_STARKNET_CONTRACT_ADDRESS", + "NEXT_PUBLIC_CANVAS_NFT_CONTRACT_ADDRESS", + "NEXT_PUBLIC_USERNAME_STORE_CONTRACT_ADDRESS", + "NEXT_PUBLIC_ART_PEACE_CONTRACT_ADDRESS", + "EXPO_PUBLIC_STARKNET_CONTRACT_ADDRESS", + "EXPO_PUBLIC_CANVAS_NFT_CONTRACT_ADDRESS", + "EXPO_PUBLIC_USERNAME_STORE_CONTRACT_ADDRESS", + "EXPO_PUBLIC_ART_PEACE_CONTRACT_ADDRESS", "BACKEND_DATABASE_URL", "CLOUDINARY_CLOUD_NAME", "CLOUDINARY_API_KEY",