Skip to content

Commit

Permalink
Merge pull request #1884 from trilitech/improve-multisig-fetching
Browse files Browse the repository at this point in the history
Improve multisig account fetching
  • Loading branch information
serjonya-trili authored Sep 13, 2024
2 parents ac8a6b9 + e92c788 commit 18a8331
Show file tree
Hide file tree
Showing 4 changed files with 132 additions and 55 deletions.
126 changes: 89 additions & 37 deletions packages/multisig/src/fetch.test.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import * as api from "@tzkt/sdk-api";
import { DefaultNetworks, GHOSTNET, mockImplicitAddress } from "@umami/tezos";
import axios from "axios";
import range from "lodash/range";

import {
CODE_HASH,
Expand All @@ -13,43 +14,9 @@ import {
const mockedAxios = jest.spyOn(axios, "get");

const mockedContractsGet = jest.spyOn(api, "contractsGet");
const mockedContractsGetCount = jest.spyOn(api, "contractsGetCount");

const multisigContracts = [
{
id: 533705,
type: "contract",
address: "KT1Mqvf7bnYe4Ty2n7ZbGkdbebCd4WoTJUUp",
kind: "smart_contract",
balance: 0,
creator: { address: "tz1LbSsDSmekew3prdDGx1nS22ie6jjBN6B3" },
numContracts: 0,
activeTokensCount: 0,
tokensCount: 0,
tokenBalancesCount: 0,
tokenTransfersCount: 0,
numDelegations: 0,
numOriginations: 1,
numTransactions: 0,
numReveals: 0,
numMigrations: 0,
transferTicketCount: 0,
increasePaidStorageCount: 0,
eventsCount: 0,
firstActivity: 1636117,
firstActivityTime: "2022-12-09T16:49:25Z",
lastActivity: 1636117,
lastActivityTime: "2022-12-09T16:49:25Z",
storage: {
owner: "tz1LbSsDSmekew3prdDGx1nS22ie6jjBN6B3",
signers: ["tz1LbSsDSmekew3prdDGx1nS22ie6jjBN6B3", "tz1dyX3B1CFYa2DfdFLyPtiJCfQRUgPVME6E"],
metadata: 216412,
threshold: "1",
last_op_id: "0",
pending_ops: 216411,
},
typeHash: 1963879877,
codeHash: -1890025422,
},
{
id: 536908,
type: "contract",
Expand Down Expand Up @@ -124,11 +91,47 @@ const multisigContracts = [
typeHash: 1963879877,
codeHash: -1890025422,
},
{
id: 533705,
type: "contract",
address: "KT1Mqvf7bnYe4Ty2n7ZbGkdbebCd4WoTJUUp",
kind: "smart_contract",
balance: 0,
creator: { address: "tz1LbSsDSmekew3prdDGx1nS22ie6jjBN6B3" },
numContracts: 0,
activeTokensCount: 0,
tokensCount: 0,
tokenBalancesCount: 0,
tokenTransfersCount: 0,
numDelegations: 0,
numOriginations: 1,
numTransactions: 0,
numReveals: 0,
numMigrations: 0,
transferTicketCount: 0,
increasePaidStorageCount: 0,
eventsCount: 0,
firstActivity: 1636117,
firstActivityTime: "2022-12-09T16:49:25Z",
lastActivity: 1636117,
lastActivityTime: "2022-12-09T16:49:25Z",
storage: {
owner: "tz1LbSsDSmekew3prdDGx1nS22ie6jjBN6B3",
signers: ["tz1LbSsDSmekew3prdDGx1nS22ie6jjBN6B3", "tz1dyX3B1CFYa2DfdFLyPtiJCfQRUgPVME6E"],
metadata: 216412,
threshold: "1",
last_op_id: "0",
pending_ops: 216411,
},
typeHash: 1963879877,
codeHash: -1890025422,
},
];

describe("multisig fetch", () => {
const expectedMockedMultisigContracts = [
{
id: 533705,
address: "KT1Mqvf7bnYe4Ty2n7ZbGkdbebCd4WoTJUUp",
storage: {
pending_ops: 216411,
Expand All @@ -137,6 +140,7 @@ describe("multisig fetch", () => {
},
},
{
id: 536908,
address: "KT1VwWbTMRN5uX4bfxCcpJnPP6iAhboqhGZr",
storage: {
pending_ops: 219458,
Expand All @@ -149,6 +153,7 @@ describe("multisig fetch", () => {
},
},
{
id: 537023,
address: "KT1Vdhz4izz7LASWU4tTLu3GBsvhJ8ULSi3G",
storage: {
pending_ops: 219535,
Expand All @@ -160,26 +165,73 @@ describe("multisig fetch", () => {

describe("getAllMultiSigContracts", () => {
it("fetches all multisig contracts", async () => {
mockedContractsGet.mockResolvedValue(multisigContracts as any);
mockedContractsGetCount.mockResolvedValue(multisigContracts.length);
mockedContractsGet.mockResolvedValue(multisigContracts);
const result = await getAllMultisigContracts(GHOSTNET);

expect(mockedContractsGetCount).toHaveBeenCalledTimes(1);
expect(mockedContractsGetCount).toHaveBeenCalledWith(
{
kind: { eq: "smart_contract" },
typeHash: { eq: TYPE_HASH },
codeHash: { eq: CODE_HASH },
},
{ baseUrl: GHOSTNET.tzktApiUrl }
);
expect(mockedContractsGet).toHaveBeenCalledTimes(1);
expect(mockedContractsGet).toHaveBeenCalledWith(
{
kind: { eq: "smart_contract" },
typeHash: { eq: TYPE_HASH },
codeHash: { eq: CODE_HASH },
select: { fields: ["id,address,storage"] },
includeStorage: true,
limit: 10000,
offset: { pg: 0 },
},
{ baseUrl: GHOSTNET.tzktApiUrl }
);
expect(
result.map(({ address, storage: { pending_ops, signers, threshold } }) => ({
result.map(({ id, address, storage: { pending_ops, signers, threshold } }) => ({
id,
address,
storage: { pending_ops, signers, threshold },
}))
).toEqual(expectedMockedMultisigContracts);
});

it("handles pagination", async () => {
mockedContractsGetCount.mockResolvedValue(79123);
mockedContractsGet.mockResolvedValue([]);

await getAllMultisigContracts(GHOSTNET);

expect(mockedContractsGetCount).toHaveBeenCalledTimes(1);
expect(mockedContractsGetCount).toHaveBeenCalledWith(
{
kind: { eq: "smart_contract" },
typeHash: { eq: TYPE_HASH },
codeHash: { eq: CODE_HASH },
},
{ baseUrl: GHOSTNET.tzktApiUrl }
);

expect(mockedContractsGet).toHaveBeenCalledTimes(8);
range(8).forEach(index => {
expect(mockedContractsGet).toHaveBeenCalledWith(
{
kind: { eq: "smart_contract" },
typeHash: { eq: TYPE_HASH },
codeHash: { eq: CODE_HASH },
select: { fields: ["id,address,storage"] },
includeStorage: true,
limit: 10000,
offset: { pg: index },
},
{ baseUrl: GHOSTNET.tzktApiUrl }
);
});
});
});

describe("getExistingContracts", () => {
Expand Down
49 changes: 34 additions & 15 deletions packages/multisig/src/fetch.ts
Original file line number Diff line number Diff line change
@@ -1,27 +1,46 @@
import { contractsGet } from "@tzkt/sdk-api";
import { contractsGet, contractsGetCount } from "@tzkt/sdk-api";
import { type Network, type RawPkh } from "@umami/tezos";
import { withRateLimit } from "@umami/tzkt";
import axios from "axios";
import range from "lodash/range";
import sortBy from "lodash/sortBy";

import { type RawTzktMultisigBigMap, type RawTzktMultisigContract } from "./types";

export const TYPE_HASH = 1963879877;
export const CODE_HASH = -1890025422;
const MULTISIG_QUERY = {
kind: { eq: "smart_contract" as const },
typeHash: { eq: TYPE_HASH },
codeHash: { eq: CODE_HASH },
};
const LIMIT = 10000;

export const getAllMultisigContracts = async (network: Network) =>
withRateLimit(() =>
contractsGet(
{
typeHash: { eq: TYPE_HASH },
codeHash: { eq: CODE_HASH },
includeStorage: true,
limit: 10000,
},
{
baseUrl: network.tzktApiUrl,
}
export const getAllMultisigContracts = async (network: Network) => {
const count = await withRateLimit(() =>
contractsGetCount(MULTISIG_QUERY as any, { baseUrl: network.tzktApiUrl })
);

const batches = await Promise.all(
range(Math.ceil(count / LIMIT)).map(index =>
withRateLimit(() =>
contractsGet(
{
...MULTISIG_QUERY,
includeStorage: true,
limit: LIMIT,
offset: { pg: index },
select: { fields: [["id", "address", "storage"].join(",")] },
},
{
baseUrl: network.tzktApiUrl,
}
)
)
)
) as Promise<RawTzktMultisigContract[]>;
);
return sortBy(batches.flat(), "id") as RawTzktMultisigContract[];
};

/**
* Returns existing addresses for the given contract addresses list & the given network.
Expand All @@ -36,7 +55,7 @@ export const getExistingContracts = (pkhs: RawPkh[], network: Network): Promise<
{
address: { in: [pkhs.join(",")] },
select: { fields: ["address"] },
limit: Math.min(10000, pkhs.length),
limit: Math.min(LIMIT, pkhs.length),
},
{
baseUrl: network.tzktApiUrl,
Expand Down
11 changes: 8 additions & 3 deletions packages/multisig/src/helpers.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,18 +14,22 @@ import {
getRelevantMultisigContracts,
parseMultisig,
} from "./helpers";
import { type RawTzktMultisigContract } from "./types";

const mockedAxios = jest.spyOn(axios, "get");

const mockedContractsGet = jest.spyOn(api, "contractsGet");
const mockedContractsGetCount = jest.spyOn(api, "contractsGetCount");

const tzktGetSameMultisigsResponse: RawTzktMultisigContract[] = [
const tzktGetSameMultisigsResponse = [
{
id: 1,
type: "contract",
address: mockContractAddress(0).pkh,
storage: { threshold: "2", pending_ops: 0, signers: [mockImplicitAddress(0).pkh] },
},
{
id: 2,
type: "contract",
address: mockContractAddress(2).pkh,
storage: { threshold: "2", pending_ops: 1, signers: [mockImplicitAddress(2).pkh] },
},
Expand All @@ -35,7 +39,8 @@ describe("multisig helpers", () => {
describe.each(DefaultNetworks)("on $name", network => {
describe("getRelevantMultisigContracts", () => {
it("fetches multisig contracts", async () => {
mockedContractsGet.mockResolvedValue(tzktGetSameMultisigsResponse as any);
mockedContractsGetCount.mockResolvedValue(tzktGetSameMultisigsResponse.length);
mockedContractsGet.mockResolvedValue(tzktGetSameMultisigsResponse);

const result = await getRelevantMultisigContracts(
new Set([mockImplicitAddress(0).pkh]),
Expand Down
1 change: 1 addition & 0 deletions packages/multisig/src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ export type Multisig = {
export type MultisigPendingOperations = Record<BigmapId, MultisigOperation[] | undefined>;

export type RawTzktMultisigContract = {
id: number;
address: string;
storage: {
signers: string[];
Expand Down

1 comment on commit 18a8331

@github-actions
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Title Lines Statements Branches Functions
apps/desktop Coverage: 83%
83.79% (1758/2098) 78.94% (825/1045) 78.45% (448/571)
apps/web Coverage: 83%
83.79% (1758/2098) 78.94% (825/1045) 78.45% (448/571)
packages/components Coverage: 97%
97.1% (134/138) 96.49% (55/57) 82.92% (34/41)
packages/core Coverage: 82%
82.89% (223/269) 73.18% (101/138) 81.35% (48/59)
packages/crypto Coverage: 100%
100% (28/28) 100% (3/3) 100% (5/5)
packages/data-polling Coverage: 98%
96.55% (140/145) 95.45% (21/22) 92.85% (39/42)
packages/multisig Coverage: 98%
98.47% (129/131) 89.47% (17/19) 100% (35/35)
packages/social-auth Coverage: 100%
100% (21/21) 100% (11/11) 100% (3/3)
packages/state Coverage: 84%
83.64% (772/923) 80.78% (164/203) 78.22% (291/372)
packages/tezos Coverage: 86%
85.57% (89/104) 89.47% (17/19) 82.75% (24/29)
packages/tzkt Coverage: 86%
84.05% (58/69) 81.25% (13/16) 76.92% (30/39)

Please sign in to comment.