Compare commits
11 Commits
buy-refact
...
develop
Author | SHA1 | Date | |
---|---|---|---|
|
a24ee193d4 | ||
|
9b325ac917 | ||
|
c3d770f713 | ||
|
3ef1694217 | ||
|
2b707e81c2 | ||
|
f6a9ab854c | ||
|
474af2fbfc | ||
|
4af059f6b7 | ||
|
23163be99d | ||
|
b956c8ec2b | ||
|
1d429f039a |
24
src/App.vue
24
src/App.vue
@ -5,8 +5,8 @@ import SpinnerComponent from "@/components/SpinnerComponent.vue";
|
|||||||
import ToasterComponent from "@/components/ToasterComponent.vue";
|
import ToasterComponent from "@/components/ToasterComponent.vue";
|
||||||
import { init, useOnboard } from "@web3-onboard/vue";
|
import { init, useOnboard } from "@web3-onboard/vue";
|
||||||
import injectedModule from "@web3-onboard/injected-wallets";
|
import injectedModule from "@web3-onboard/injected-wallets";
|
||||||
import { Networks } from "./model/Networks";
|
import { Networks } from "@/model/Networks";
|
||||||
import { NetworkEnum } from "./model/NetworkEnum";
|
import { NetworkEnum } from "@/model/NetworkEnum";
|
||||||
import { ref } from "vue";
|
import { ref } from "vue";
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
@ -15,20 +15,12 @@ const targetNetwork = ref(NetworkEnum.sepolia);
|
|||||||
|
|
||||||
const web3Onboard = init({
|
const web3Onboard = init({
|
||||||
wallets: [injected],
|
wallets: [injected],
|
||||||
chains: [
|
chains: Object.entries(Networks).map(([, network]) => ({
|
||||||
{
|
id: network.chainId,
|
||||||
id: Networks[NetworkEnum.sepolia].chainId,
|
token: network.token,
|
||||||
token: "ETH",
|
label: network.chainName,
|
||||||
label: "Sepolia",
|
rpcUrl: network.rpcUrl,
|
||||||
rpcUrl: import.meta.env.VITE_SEPOLIA_API_URL,
|
})),
|
||||||
},
|
|
||||||
{
|
|
||||||
id: Networks[NetworkEnum.rootstock].chainId,
|
|
||||||
token: "tRBTC",
|
|
||||||
label: "Rootstock Testnet",
|
|
||||||
rpcUrl: import.meta.env.VITE_ROOTSTOCK_API_URL,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
connect: {
|
connect: {
|
||||||
autoConnectLastWallet: true,
|
autoConnectLastWallet: true,
|
||||||
},
|
},
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { useUser } from "@/composables/useUser";
|
import { useUser } from "@/composables/useUser";
|
||||||
import { NetworkEnum, TokenEnum } from "@/model/NetworkEnum";
|
import { NetworkEnum, TokenEnum } from "@/model/NetworkEnum";
|
||||||
import { createPublicClient, http, type Address } from "viem";
|
import { createPublicClient, http, type Address } from "viem";
|
||||||
import { sepolia, rootstock } from "viem/chains";
|
import { sepolia, rootstockTestnet } from "viem/chains";
|
||||||
|
|
||||||
const Tokens: { [key in NetworkEnum]: { [key in TokenEnum]: Address } } = {
|
const Tokens: { [key in NetworkEnum]: { [key in TokenEnum]: Address } } = {
|
||||||
[NetworkEnum.sepolia]: {
|
[NetworkEnum.sepolia]: {
|
||||||
@ -39,7 +39,7 @@ export const getP2PixAddress = (network?: NetworkEnum): Address => {
|
|||||||
const user = useUser();
|
const user = useUser();
|
||||||
const possibleP2PixAddresses: { [key in NetworkEnum]: Address } = {
|
const possibleP2PixAddresses: { [key in NetworkEnum]: Address } = {
|
||||||
[NetworkEnum.sepolia]: "0xb7cD135F5eFD9760981e02E2a898790b688939fe",
|
[NetworkEnum.sepolia]: "0xb7cD135F5eFD9760981e02E2a898790b688939fe",
|
||||||
[NetworkEnum.rootstock]: "0x98ba35eb14b38D6Aa709338283af3e922476dE34",
|
[NetworkEnum.rootstock]: "0x57Dcba05980761169508886eEdc6f5E7EC0411Dc",
|
||||||
};
|
};
|
||||||
|
|
||||||
return possibleP2PixAddresses[
|
return possibleP2PixAddresses[
|
||||||
@ -58,7 +58,7 @@ export const getProviderUrl = (network?: NetworkEnum): string => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const getProviderByNetwork = (network: NetworkEnum) => {
|
export const getProviderByNetwork = (network: NetworkEnum) => {
|
||||||
const chain = network === NetworkEnum.sepolia ? sepolia : rootstock;
|
const chain = network === NetworkEnum.sepolia ? sepolia : rootstockTestnet;
|
||||||
return createPublicClient({
|
return createPublicClient({
|
||||||
chain,
|
chain,
|
||||||
transport: http(getProviderUrl(network)),
|
transport: http(getProviderUrl(network)),
|
||||||
|
@ -1,15 +1,9 @@
|
|||||||
import { getContract } from "./provider";
|
import { getContract } from "./provider";
|
||||||
import { getTokenAddress } from "./addresses";
|
import { getTokenAddress } from "./addresses";
|
||||||
import {
|
import {
|
||||||
bytesToHex,
|
|
||||||
encodeAbiParameters,
|
|
||||||
keccak256,
|
|
||||||
parseAbiParameters,
|
|
||||||
parseEther,
|
parseEther,
|
||||||
stringToBytes,
|
|
||||||
stringToHex,
|
|
||||||
toBytes,
|
|
||||||
type Address,
|
type Address,
|
||||||
|
type TransactionReceipt,
|
||||||
} from "viem";
|
} from "viem";
|
||||||
import type { TokenEnum } from "@/model/NetworkEnum";
|
import type { TokenEnum } from "@/model/NetworkEnum";
|
||||||
|
|
||||||
@ -69,24 +63,20 @@ export const withdrawDeposit = async (
|
|||||||
|
|
||||||
export const releaseLock = async (
|
export const releaseLock = async (
|
||||||
lockID: bigint,
|
lockID: bigint,
|
||||||
pixtarget: string,
|
pixTimestamp: `0x${string}`&{lenght:34},
|
||||||
signature: string
|
signature: `0x${string}`
|
||||||
): Promise<any> => {
|
): Promise<TransactionReceipt> => {
|
||||||
const { address, abi, wallet, client, account } = await getContract();
|
const { address, abi, wallet, client, account } = await getContract();
|
||||||
|
|
||||||
console.log("Releasing lock", { lockID, pixtarget, signature });
|
|
||||||
if (!wallet) {
|
if (!wallet) {
|
||||||
throw new Error("Wallet not connected");
|
throw new Error("Wallet not connected");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert pixtarget to bytes32
|
|
||||||
const pixTimestamp = keccak256(stringToHex(pixtarget, { size: 32 }) );
|
|
||||||
|
|
||||||
const { request } = await client.simulateContract({
|
const { request } = await client.simulateContract({
|
||||||
address,
|
address,
|
||||||
abi,
|
abi,
|
||||||
functionName: "release",
|
functionName: "release",
|
||||||
args: [BigInt(lockID), pixTimestamp, stringToHex(signature)],
|
args: [BigInt(lockID), pixTimestamp, signature],
|
||||||
account
|
account
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -32,8 +32,8 @@ const getNetworksLiquidity = async (): Promise<void> => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const getParticipantID = async (
|
const getParticipantID = async (
|
||||||
seller: string,
|
seller: Address,
|
||||||
token: string
|
token: Address
|
||||||
): Promise<string> => {
|
): Promise<string> => {
|
||||||
const { address, abi, client } = await getContract();
|
const { address, abi, client } = await getContract();
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ import {
|
|||||||
PublicClient,
|
PublicClient,
|
||||||
WalletClient,
|
WalletClient,
|
||||||
} from "viem";
|
} from "viem";
|
||||||
import { sepolia, rootstock } from "viem/chains";
|
import { sepolia, rootstockTestnet } from "viem/chains";
|
||||||
import { useUser } from "@/composables/useUser";
|
import { useUser } from "@/composables/useUser";
|
||||||
|
|
||||||
let walletClient: WalletClient | null = null;
|
let walletClient: WalletClient | null = null;
|
||||||
@ -19,7 +19,7 @@ const getPublicClient = (): PublicClient => {
|
|||||||
const rpcUrl = getProviderUrl();
|
const rpcUrl = getProviderUrl();
|
||||||
return createPublicClient({
|
return createPublicClient({
|
||||||
chain:
|
chain:
|
||||||
Number(user.networkName.value) === sepolia.id ? sepolia : rootstock,
|
Number(user.networkName.value) === sepolia.id ? sepolia : rootstockTestnet,
|
||||||
transport: http(rpcUrl),
|
transport: http(rpcUrl),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
@ -46,7 +46,7 @@ const getContract = async (onlyRpcProvider = false) => {
|
|||||||
const connectProvider = async (p: any): Promise<void> => {
|
const connectProvider = async (p: any): Promise<void> => {
|
||||||
const user = useUser();
|
const user = useUser();
|
||||||
const chain =
|
const chain =
|
||||||
Number(user.networkName.value) === sepolia.id ? sepolia : rootstock;
|
Number(user.networkName.value) === sepolia.id ? sepolia : rootstockTestnet;
|
||||||
|
|
||||||
const [account] = await p!.request({ method: "eth_requestAccounts" });
|
const [account] = await p!.request({ method: "eth_requestAccounts" });
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { getContract, getPublicClient, getWalletClient } from "./provider";
|
import { getContract, getPublicClient, getWalletClient } from "./provider";
|
||||||
import { getTokenAddress, getP2PixAddress } from "./addresses";
|
import { getTokenAddress, getP2PixAddress } from "./addresses";
|
||||||
import { parseEther, toHex } from "viem";
|
import { parseEther, toHex } from "viem";
|
||||||
import { sepolia, rootstock } from "viem/chains";
|
import { sepolia, rootstockTestnet } from "viem/chains";
|
||||||
|
|
||||||
import { mockTokenAbi } from "./abi";
|
import { mockTokenAbi } from "./abi";
|
||||||
import { useUser } from "@/composables/useUser";
|
import { useUser } from "@/composables/useUser";
|
||||||
@ -33,7 +33,7 @@ const approveTokens = async (participant: Participant): Promise<any> => {
|
|||||||
|
|
||||||
if ( allowance < parseEther(participant.offer.toString()) ) {
|
if ( allowance < parseEther(participant.offer.toString()) ) {
|
||||||
// Approve tokens
|
// Approve tokens
|
||||||
const chain = user.networkId.value === sepolia.id ? sepolia : rootstock;
|
const chain = user.networkId.value === sepolia.id ? sepolia : rootstockTestnet;
|
||||||
const hash = await walletClient.writeContract({
|
const hash = await walletClient.writeContract({
|
||||||
address: tokenAddress,
|
address: tokenAddress,
|
||||||
abi: mockTokenAbi,
|
abi: mockTokenAbi,
|
||||||
@ -65,7 +65,7 @@ const addDeposit = async (): Promise<any> => {
|
|||||||
if (!sellerId.id) {
|
if (!sellerId.id) {
|
||||||
throw new Error("Failed to create participant");
|
throw new Error("Failed to create participant");
|
||||||
}
|
}
|
||||||
const chain = user.networkId.value === sepolia.id ? sepolia : rootstock;
|
const chain = user.networkId.value === sepolia.id ? sepolia : rootstockTestnet;
|
||||||
const hash = await walletClient.writeContract({
|
const hash = await walletClient.writeContract({
|
||||||
address,
|
address,
|
||||||
abi,
|
abi,
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { formatEther, hexToString, type Address } from "viem";
|
import { formatEther, type Address } from "viem";
|
||||||
import { useUser } from "@/composables/useUser";
|
import { useUser } from "@/composables/useUser";
|
||||||
|
|
||||||
import { getPublicClient, getWalletClient, getContract } from "./provider";
|
import { getPublicClient, getWalletClient, getContract } from "./provider";
|
||||||
@ -463,18 +463,3 @@ export const getActiveLockAmount = async (
|
|||||||
return total;
|
return total;
|
||||||
}, 0);
|
}, 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getSellerParticipantId = async (
|
|
||||||
sellerAddress: Address,
|
|
||||||
tokenAddress: Address
|
|
||||||
): Promise<string> => {
|
|
||||||
const { address, abi, client } = await getContract();
|
|
||||||
|
|
||||||
const participantId = await client.readContract({
|
|
||||||
address,
|
|
||||||
abi,
|
|
||||||
functionName: "getPixTarget",
|
|
||||||
args: [sellerAddress, tokenAddress],
|
|
||||||
});
|
|
||||||
return hexToString(participantId);
|
|
||||||
};
|
|
||||||
|
@ -4,7 +4,7 @@ import CustomButton from "@/components/CustomButton/CustomButton.vue";
|
|||||||
import CustomModal from "@/components//CustomModal/CustomModal.vue";
|
import CustomModal from "@/components//CustomModal/CustomModal.vue";
|
||||||
import SpinnerComponent from "@/components/SpinnerComponent.vue";
|
import SpinnerComponent from "@/components/SpinnerComponent.vue";
|
||||||
import { createSolicitation, getSolicitation, type Offer } from "@/utils/bbPay";
|
import { createSolicitation, getSolicitation, type Offer } from "@/utils/bbPay";
|
||||||
import { getSellerParticipantId } from "@/blockchain/wallet";
|
import { getParticipantID } from "@/blockchain/events";
|
||||||
import { getUnreleasedLockById } from "@/blockchain/events";
|
import { getUnreleasedLockById } from "@/blockchain/events";
|
||||||
import QRCode from "qrcode";
|
import QRCode from "qrcode";
|
||||||
|
|
||||||
@ -18,7 +18,7 @@ const props = defineProps<Props>();
|
|||||||
const qrCode = ref<string>("");
|
const qrCode = ref<string>("");
|
||||||
const qrCodeSvg = ref<string>("");
|
const qrCodeSvg = ref<string>("");
|
||||||
const showWarnModal = ref<boolean>(true);
|
const showWarnModal = ref<boolean>(true);
|
||||||
const pixTarget = ref<string>("");
|
const pixTimestamp = ref<string>("");
|
||||||
const releaseSignature = ref<string>("");
|
const releaseSignature = ref<string>("");
|
||||||
const solicitationData = ref<any>(null);
|
const solicitationData = ref<any>(null);
|
||||||
const pollingInterval = ref<NodeJS.Timeout | null>(null);
|
const pollingInterval = ref<NodeJS.Timeout | null>(null);
|
||||||
@ -56,7 +56,7 @@ const checkSolicitationStatus = async () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (response.signature) {
|
if (response.signature) {
|
||||||
pixTarget.value = response.pixTarget;
|
pixTimestamp.value = response.pixTimestamp;
|
||||||
releaseSignature.value = response.signature;
|
releaseSignature.value = response.signature;
|
||||||
// Stop polling when payment is confirmed
|
// Stop polling when payment is confirmed
|
||||||
if (pollingInterval.value) {
|
if (pollingInterval.value) {
|
||||||
@ -86,7 +86,7 @@ onMounted(async () => {
|
|||||||
BigInt(props.lockID)
|
BigInt(props.lockID)
|
||||||
);
|
);
|
||||||
|
|
||||||
const participantId = await getSellerParticipantId(
|
const participantId = await getParticipantID(
|
||||||
sellerAddress,
|
sellerAddress,
|
||||||
tokenAddress
|
tokenAddress
|
||||||
);
|
);
|
||||||
@ -170,7 +170,7 @@ onUnmounted(() => {
|
|||||||
releaseSignature ? 'Enviar para a rede' : 'Validando pagamento...'
|
releaseSignature ? 'Enviar para a rede' : 'Validando pagamento...'
|
||||||
"
|
"
|
||||||
@button-clicked="
|
@button-clicked="
|
||||||
emit('pixValidated', { pixTarget, signature: releaseSignature })
|
emit('pixValidated', { pixTimestamp, signature: releaseSignature })
|
||||||
"
|
"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
@ -2,7 +2,6 @@ import { ref } from "vue";
|
|||||||
import { NetworkEnum, TokenEnum } from "../model/NetworkEnum";
|
import { NetworkEnum, TokenEnum } from "../model/NetworkEnum";
|
||||||
import type { ValidDeposit } from "@/model/ValidDeposit";
|
import type { ValidDeposit } from "@/model/ValidDeposit";
|
||||||
import type { Participant } from "../utils/bbPay";
|
import type { Participant } from "../utils/bbPay";
|
||||||
import { NetworkById } from "@/model/Networks";
|
|
||||||
import type { Address } from "viem"
|
import type { Address } from "viem"
|
||||||
|
|
||||||
const walletAddress = ref<Address | null>(null);
|
const walletAddress = ref<Address | null>(null);
|
||||||
@ -33,7 +32,7 @@ export function useUser() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const setNetworkId = (network: string | number) => {
|
const setNetworkId = (network: string | number) => {
|
||||||
networkName.value = NetworkById(network) || NetworkEnum.sepolia;
|
networkName.value = Number(network) as NetworkEnum || NetworkEnum.sepolia;
|
||||||
networkId.value = Number(network);
|
networkId.value = Number(network);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,37 +1,26 @@
|
|||||||
import { NetworkEnum } from "@/model/NetworkEnum";
|
import { NetworkEnum } from "@/model/NetworkEnum";
|
||||||
|
|
||||||
export const NetworkById = (
|
export interface NetworkConfig {
|
||||||
chainId: string | number
|
chainId: string;
|
||||||
): NetworkEnum | undefined => {
|
chainName: string;
|
||||||
const normalizedChainId =
|
token: string;
|
||||||
typeof chainId === "number" ? chainId : Number(chainId);
|
rpcUrl?: string;
|
||||||
|
blockExplorerUrl?: string;
|
||||||
|
}
|
||||||
|
|
||||||
for (const [network, details] of Object.entries(Networks)) {
|
export const Networks: { [key in NetworkEnum]: NetworkConfig } = {
|
||||||
if (Number(details.chainId) === normalizedChainId) {
|
|
||||||
return network as unknown as NetworkEnum;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return undefined;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const Networks = {
|
|
||||||
[NetworkEnum.sepolia]: {
|
[NetworkEnum.sepolia]: {
|
||||||
chainId: "0xAA36A7",
|
chainId: "0xAA36A7",
|
||||||
chainName: "Sepolia Testnet",
|
chainName: "Sepolia Testnet",
|
||||||
|
token: "ETH",
|
||||||
|
rpcUrl: import.meta.env.VITE_SEPOLIA_API_URL,
|
||||||
|
blockExplorerUrl: "https://sepolia.etherscan.io",
|
||||||
},
|
},
|
||||||
[NetworkEnum.rootstock]: {
|
[NetworkEnum.rootstock]: {
|
||||||
chainId: "0x1F",
|
chainId: "0x1F",
|
||||||
chainName: "Rootstock Testnet",
|
chainName: "Rootstock Testnet",
|
||||||
rpcUrls: ["https://public-node.testnet.rsk.co/"],
|
token: "tRBTC",
|
||||||
iconUrls: [
|
rpcUrl: import.meta.env.VITE_ROOTSTOCK_API_URL,
|
||||||
"",
|
blockExplorerUrl: "https://explorer.testnet.rsk.co",
|
||||||
],
|
|
||||||
nativeCurrency: {
|
|
||||||
name: "tRBTC",
|
|
||||||
symbol: "tRBTC",
|
|
||||||
decimals: 18,
|
|
||||||
},
|
|
||||||
blockExplorerUrls: ["https://explorer.testnet.rootstock.io/"],
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -61,15 +61,15 @@ export const createSolicitation = async (offer: Offer) => {
|
|||||||
return response.json();
|
return response.json();
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getSolicitation = async (id: string) => {
|
export const getSolicitation = async (id: bigint): Promise<{pixTimestamp: `0x${string}`, signature: `0x${string}`}> => {
|
||||||
const response = await fetch(
|
const response = await fetch(
|
||||||
`${import.meta.env.VITE_APP_API_URL}/release/${id}`
|
`${import.meta.env.VITE_APP_API_URL}/release/${id}`
|
||||||
);
|
);
|
||||||
|
|
||||||
const obj: any = await response.json();
|
const obj = await response.json();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
pixTarget: obj.pixTarget,
|
pixTimestamp: obj.pixTimestamp,
|
||||||
signature: obj.signature,
|
signature: obj.signature,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -59,19 +59,15 @@ const confirmBuyClick = async (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const releaseTransaction = async ({
|
const releaseTransaction = async (params: {
|
||||||
pixTarget,
|
pixTimestamp: `0x${string}`&{lenght:34},
|
||||||
signature,
|
signature: `0x${string}`,
|
||||||
}: {
|
|
||||||
pixTarget: string;
|
|
||||||
signature: string;
|
|
||||||
}) => {
|
}) => {
|
||||||
flowStep.value = Step.List;
|
flowStep.value = Step.List;
|
||||||
showBuyAlert.value = true;
|
showBuyAlert.value = true;
|
||||||
loadingRelease.value = true;
|
loadingRelease.value = true;
|
||||||
|
|
||||||
const release = await releaseLock(BigInt(lockID.value), pixTarget, signature);
|
const release = await releaseLock(BigInt(lockID.value), params.pixTimestamp, params.signature);
|
||||||
await release.wait();
|
|
||||||
|
|
||||||
await updateWalletStatus();
|
await updateWalletStatus();
|
||||||
loadingRelease.value = false;
|
loadingRelease.value = false;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user