Bläddra i källkod

trying out 1 price feed test case

Ayush Suresh 3 månader sedan
förälder
incheckning
f9c0550630

+ 52 - 32
target_chains/ton/contracts/contracts/common/utils.fc

@@ -77,46 +77,66 @@ Note:
    - If the input data exceeds 1016 bits, it is split into multiple cells linked by references
    - Uses direct forward construction to avoid gas-inefficient double reversal
 -}
-(cell, slice) read_and_store_large_data(slice in_msg_body, int size) {
-    ;; Collect chunks in order as we build them
-    tuple chunk_list = empty_tuple();
-    int total_bits_loaded = 0;
-    builder current_builder = begin_cell();
+;; (cell, slice) read_and_store_large_data(slice in_msg_body, int size) {
+;;     ;; Collect chunks in order as we build them
+;;     tuple chunk_list = empty_tuple();
+;;     int total_bits_loaded = 0;
+;;     builder current_builder = begin_cell();
 
-    while ((~ in_msg_body.slice_empty?()) & (total_bits_loaded < size)) {
-        int bits_to_load = min(min(in_msg_body.slice_bits(), MAX_BITS - current_builder.builder_bits()), size - total_bits_loaded);
-        current_builder = current_builder.store_slice(in_msg_body~load_bits(bits_to_load));
-        total_bits_loaded += bits_to_load;
+;;     while ((~ in_msg_body.slice_empty?()) & (total_bits_loaded < size)) {
+;;         int bits_to_load = min(min(in_msg_body.slice_bits(), MAX_BITS - current_builder.builder_bits()), size - total_bits_loaded);
+;;         current_builder = current_builder.store_slice(in_msg_body~load_bits(bits_to_load));
+;;         total_bits_loaded += bits_to_load;
         
-        if ((current_builder.builder_bits() == MAX_BITS) | (size - total_bits_loaded == 0)) {
-            cell current_chunk = current_builder.end_cell();
-            chunk_list~tpush(current_chunk);
-            current_builder = begin_cell();
-        }
+;;         if ((current_builder.builder_bits() == MAX_BITS) | (size - total_bits_loaded == 0)) {
+;;             cell current_chunk = current_builder.end_cell();
+;;             chunk_list~tpush(current_chunk);
+;;             current_builder = begin_cell();
+;;         }
         
-        if ((in_msg_body.slice_bits() == 0) & (~ in_msg_body.slice_refs_empty?())) {
-            in_msg_body = in_msg_body~load_ref().begin_parse();
-        }
-    }
+;;         if ((in_msg_body.slice_bits() == 0) & (~ in_msg_body.slice_refs_empty?())) {
+;;             in_msg_body = in_msg_body~load_ref().begin_parse();
+;;         }
+;;     }
     
-    ;; Build forward chain: first chunk → second chunk → third chunk etc
-    cell result = null();
-    int chunk_count = chunk_list.tlen();
+;;     ;; Build forward chain: first chunk → second chunk → third chunk etc
+;;     cell result = null();
+;;     int chunk_count = chunk_list.tlen();
     
-    if (chunk_count > 0) {
-        ;; Start from the last chunk (no references)
-        result = chunk_list.at(chunk_count - 1);
+;;     if (chunk_count > 0) {
+;;         ;; Start from the last chunk (no references)
+;;         result = chunk_list.at(chunk_count - 1);
         
-        ;; Build forward by adding references from earlier chunks to later chunks
-        int i = chunk_count - 2;
-        while (i >= 0) {
-            cell current_chunk = chunk_list.at(i);
-            result = begin_cell().store_slice(current_chunk.begin_parse()).store_ref(result).end_cell();
-            i -= 1;
+;;         ;; Build forward by adding references from earlier chunks to later chunks
+;;         int i = chunk_count - 2;
+;;         while (i >= 0) {
+;;             cell current_chunk = chunk_list.at(i);
+;;             result = begin_cell().store_slice(current_chunk.begin_parse()).store_ref(result).end_cell();
+;;             i -= 1;
+;;         }
+;;     }
+    
+;;     return (result, in_msg_body);
+;; }
+
+(cell, slice) read_and_store_large_data(slice in_msg_body, int size) {
+    (cell chunks, slice remaining) = split_into_reverse_chunks(in_msg_body, size);
+    cell last_cell = null();
+    while (~ cell_null?(chunks)) {
+        slice chunk = chunks.begin_parse();
+        builder cb = begin_cell().store_slice(chunk~load_bits(chunk.slice_bits()));
+        if (~ cell_null?(last_cell)) {
+            cb = cb.store_ref(last_cell);
+        }
+        last_cell = cb.end_cell();
+        if (chunk.slice_refs_empty?()) {
+            chunks = null();
+        } else {
+            chunks = chunk~load_ref();
         }
     }
-    
-    return (result, in_msg_body);
+
+    return (last_cell, remaining);
 }
 
 (int) pubkey_to_eth_address(int x1, int x2) {

+ 426 - 26
target_chains/ton/contracts/tests/PythTest.spec.ts

@@ -6,15 +6,6 @@ import { HexString, Price } from "@pythnetwork/price-service-sdk";
 import { PythTest, PythTestConfig } from "../wrappers/PythTest";
 import {
   BTC_PRICE_FEED_ID,
-  DOGE_PRICE_FEED_ID,
-  SOL_USD_PRICE_FEED_ID,
-  PYTH_USD_PRICE_FEED_ID,
-  ARBITRUM_USD_PRICE_FEED_ID,
-  TON_USD_PRICE_FEED_ID,
-  AAPL_PRICE_FEED_ID,
-  ABNB_PRICE_FEED_ID,
-  ADBE_PRICE_FEED_ID,
-  AMZN_PRICE_FEED_ID,
   HERMES_BTC_ETH_UPDATE,
   PYTH_SET_DATA_SOURCES,
   PYTH_SET_FEE,
@@ -55,12 +46,176 @@ import {
   HERMES_ETH_UNIQUE_PRICE,
   HERMES_ETH_UNIQUE_PUBLISH_TIME,
   HERMES_UPDATE_10_PRICE_FEEDS,
+  DOGE_PRICE_FEED_ID,
+  SOL_USD_PRICE_FEED_ID,
+  PYTH_USD_PRICE_FEED_ID,
+  ARBITRUM_USD_PRICE_FEED_ID,
+  TON_USD_PRICE_FEED_ID,
+  AAPL_PRICE_FEED_ID,
+  ABNB_PRICE_FEED_ID,
+  ADBE_PRICE_FEED_ID,
+  AMZN_PRICE_FEED_ID,
+  HERMES_10_BTC_PRICE,
+  HERMES_10_BTC_CONF,
+  HERMES_10_BTC_EXPO,
+  HERMES_10_BTC_PUBLISH_TIME,
+  HERMES_10_BTC_EMA_PRICE,
+  HERMES_10_BTC_EMA_CONF,
+  HERMES_10_BTC_EMA_EXPO,
+  HERMES_10_BTC_EMA_PUBLISH_TIME,
+  HERMES_10_ETH_PRICE,
+  HERMES_10_ETH_CONF,
+  HERMES_10_ETH_EXPO,
+  HERMES_10_ETH_PUBLISH_TIME,
+  HERMES_10_ETH_EMA_PRICE,
+  HERMES_10_ETH_EMA_CONF,
+  HERMES_10_ETH_EMA_EXPO,
+  HERMES_10_ETH_EMA_PUBLISH_TIME,
+  HERMES_10_DOGE_PRICE,
+  HERMES_10_DOGE_CONF,
+  HERMES_10_DOGE_EXPO,
+  HERMES_10_DOGE_PUBLISH_TIME,
+  HERMES_10_DOGE_EMA_PRICE,
+  HERMES_10_DOGE_EMA_CONF,
+  HERMES_10_DOGE_EMA_EXPO,
+  HERMES_10_DOGE_EMA_PUBLISH_TIME,
+  HERMES_10_SOL_PRICE,
+  HERMES_10_SOL_CONF,
+  HERMES_10_SOL_EXPO,
+  HERMES_10_SOL_PUBLISH_TIME,
+  HERMES_10_SOL_EMA_PRICE,
+  HERMES_10_SOL_EMA_CONF,
+  HERMES_10_SOL_EMA_EXPO,
+  HERMES_10_SOL_EMA_PUBLISH_TIME,
+  HERMES_10_PYTH_PRICE,
+  HERMES_10_PYTH_CONF,
+  HERMES_10_PYTH_EXPO,
+  HERMES_10_PYTH_PUBLISH_TIME,
+  HERMES_10_PYTH_EMA_PRICE,
+  HERMES_10_PYTH_EMA_CONF,
+  HERMES_10_PYTH_EMA_EXPO,
+  HERMES_10_PYTH_EMA_PUBLISH_TIME,
+  HERMES_10_ARB_PRICE,
+  HERMES_10_ARB_CONF,
+  HERMES_10_ARB_EXPO,
+  HERMES_10_ARB_PUBLISH_TIME,
+  HERMES_10_ARB_EMA_PRICE,
+  HERMES_10_ARB_EMA_CONF,
+  HERMES_10_ARB_EMA_EXPO,
+  HERMES_10_ARB_EMA_PUBLISH_TIME,
+  HERMES_10_TON_PRICE,
+  HERMES_10_TON_CONF,
+  HERMES_10_TON_EXPO,
+  HERMES_10_TON_PUBLISH_TIME,
+  HERMES_10_TON_EMA_PRICE,
+  HERMES_10_TON_EMA_CONF,
+  HERMES_10_TON_EMA_EXPO,
+  HERMES_10_TON_EMA_PUBLISH_TIME,
+  HERMES_10_AAPL_PRICE,
+  HERMES_10_AAPL_CONF,
+  HERMES_10_AAPL_EXPO,
+  HERMES_10_AAPL_PUBLISH_TIME,
+  HERMES_10_AAPL_EMA_PRICE,
+  HERMES_10_AAPL_EMA_CONF,
+  HERMES_10_AAPL_EMA_EXPO,
+  HERMES_10_AAPL_EMA_PUBLISH_TIME,
+  HERMES_10_ABNB_PRICE,
+  HERMES_10_ABNB_CONF,
+  HERMES_10_ABNB_EXPO,
+  HERMES_10_ABNB_PUBLISH_TIME,
+  HERMES_10_ABNB_EMA_PRICE,
+  HERMES_10_ABNB_EMA_CONF,
+  HERMES_10_ABNB_EMA_EXPO,
+  HERMES_10_ABNB_EMA_PUBLISH_TIME,
+  HERMES_10_ADBE_PRICE,
+  HERMES_10_ADBE_CONF,
+  HERMES_10_ADBE_EXPO,
+  HERMES_10_ADBE_PUBLISH_TIME,
+  HERMES_10_ADBE_EMA_PRICE,
+  HERMES_10_ADBE_EMA_CONF,
+  HERMES_10_ADBE_EMA_EXPO,
+  HERMES_10_ADBE_EMA_PUBLISH_TIME,
+  HERMES_10_AMZN_PRICE,
+  HERMES_10_AMZN_CONF,
+  HERMES_10_AMZN_EXPO,
+  HERMES_10_AMZN_PUBLISH_TIME,
+  HERMES_10_AMZN_EMA_PRICE,
+  HERMES_10_AMZN_EMA_CONF,
+  HERMES_10_AMZN_EMA_EXPO,
+  HERMES_10_AMZN_EMA_PUBLISH_TIME,
+  HERMES_7_BTC_PRICE,
+  HERMES_7_BTC_CONF,
+  HERMES_7_BTC_EXPO,
+  HERMES_7_BTC_PUBLISH_TIME,
+  HERMES_7_BTC_EMA_PRICE,
+  HERMES_7_BTC_EMA_CONF,
+  HERMES_7_BTC_EMA_EXPO,
+  HERMES_7_BTC_EMA_PUBLISH_TIME,
+  HERMES_7_ETH_PRICE,
+  HERMES_7_ETH_CONF,
+  HERMES_7_ETH_EXPO,
+  HERMES_7_ETH_PUBLISH_TIME,
+  HERMES_7_ETH_EMA_PRICE,
+  HERMES_7_ETH_EMA_CONF,
+  HERMES_7_ETH_EMA_EXPO,
+  HERMES_7_ETH_EMA_PUBLISH_TIME,
+  HERMES_7_DOGE_PRICE,
+  HERMES_7_DOGE_CONF,
+  HERMES_7_DOGE_EXPO,
+  HERMES_7_DOGE_PUBLISH_TIME,
+  HERMES_7_DOGE_EMA_PRICE,
+  HERMES_7_DOGE_EMA_CONF,
+  HERMES_7_DOGE_EMA_EXPO,
+  HERMES_7_DOGE_EMA_PUBLISH_TIME,
+  HERMES_7_SOL_PRICE,
+  HERMES_7_SOL_CONF,
+  HERMES_7_SOL_EXPO,
+  HERMES_7_SOL_PUBLISH_TIME,
+  HERMES_7_SOL_EMA_PRICE,
+  HERMES_7_SOL_EMA_CONF,
+  HERMES_7_SOL_EMA_EXPO,
+  HERMES_7_SOL_EMA_PUBLISH_TIME,
+  HERMES_7_PYTH_PRICE,
+  HERMES_7_PYTH_CONF,
+  HERMES_7_PYTH_EXPO,
+  HERMES_7_PYTH_PUBLISH_TIME,
+  HERMES_7_PYTH_EMA_PRICE,
+  HERMES_7_PYTH_EMA_CONF,
+  HERMES_7_PYTH_EMA_EXPO,
+  HERMES_7_PYTH_EMA_PUBLISH_TIME,
+  HERMES_7_ARB_PRICE,
+  HERMES_7_ARB_CONF,
+  HERMES_7_ARB_EXPO,
+  HERMES_7_ARB_PUBLISH_TIME,
+  HERMES_7_ARB_EMA_PRICE,
+  HERMES_7_ARB_EMA_CONF,
+  HERMES_7_ARB_EMA_EXPO,
+  HERMES_7_ARB_EMA_PUBLISH_TIME,
+  HERMES_7_TON_PRICE,
+  HERMES_7_TON_CONF,
+  HERMES_7_TON_EXPO,
+  HERMES_7_TON_PUBLISH_TIME,
+  HERMES_7_TON_EMA_PRICE,
+  HERMES_7_TON_EMA_CONF,
+  HERMES_7_TON_EMA_EXPO,
+  HERMES_7_TON_EMA_PUBLISH_TIME,
+  HERMES_UPDATE_7_PRICE_FEEDS,
+  HERMES_UPDATE_1_PRICE_FEED,
+  HERMES_1_BTC_PUBLISH_TIME,
+  HERMES_1_BTC_PRICE,
+  HERMES_1_BTC_CONF,
+  HERMES_1_BTC_EXPO,
+  HERMES_1_BTC_EMA_PRICE,
+  HERMES_1_BTC_EMA_CONF,
+  HERMES_1_BTC_EMA_EXPO,
+  HERMES_1_BTC_EMA_PUBLISH_TIME,
 } from "./utils/pyth";
 import { GUARDIAN_SET_0, MAINNET_UPGRADE_VAAS } from "./utils/wormhole";
 import { DataSource } from "@pythnetwork/xc-admin-common";
 import { createAuthorizeUpgradePayload } from "./utils";
 import {
   UniversalAddress,
+  composeLiteral,
   createVAA,
   serialize,
 } from "@wormhole-foundation/sdk-definitions";
@@ -1605,11 +1760,220 @@ describe("PythTest", () => {
     expect(btcCs.remainingRefs).toBe(0);
   });
 
-  it ("should successfully parse price feed updates with 10 price feeds", async () => {
+  it("should successfully parse price feed updates with 1 price feed", async () => {
     await deployContract();
     await updateGuardianSets(pythTest, deployer);
 
-    const sentValue = toNano("1");
+    const sentValue = toNano("15");
+    const result = await pythTest.sendParsePriceFeedUpdates(
+      deployer.getSender(),
+      Buffer.from(HERMES_UPDATE_1_PRICE_FEED, "hex"),
+      sentValue,
+      [
+        BTC_PRICE_FEED_ID,
+      ],
+      HERMES_1_BTC_PUBLISH_TIME,
+      HERMES_1_BTC_PUBLISH_TIME,
+      mockDeployer.address,
+      CUSTOM_PAYLOAD,
+    );
+
+    expect(result.transactions).toHaveTransaction({
+      from: deployer.address,
+      to: pythTest.address,
+      success: true,
+      outMessagesCount: 1,
+    });
+
+    expect(result.transactions).toHaveTransaction({
+      from: pythTest.address,
+      to: deployer.address,
+      success: true,
+    });
+
+    const outMessage = result.transactions[1].outMessages.values()[0];
+
+    expect(
+      (outMessage.info as CommonMessageInfoInternal).value.coins,
+    ).toBeGreaterThan(0);
+
+    const cs = outMessage.body.beginParse();
+
+    console.log(outMessage.body);
+
+    const op = cs.loadUint(32);
+    expect(op).toBe(5);
+
+    const numPriceFeeds = cs.loadUint(8);
+    expect(numPriceFeeds).toBe(10);
+
+    const priceFeedsCell = cs.loadRef();
+    let currentCell = priceFeedsCell;
+
+    const expectedFeeds = [
+      { id: BTC_PRICE_FEED_ID, price: HERMES_1_BTC_PRICE, conf: HERMES_1_BTC_CONF, expo: HERMES_1_BTC_EXPO, publishTime: HERMES_1_BTC_PUBLISH_TIME, emaPrice: HERMES_1_BTC_EMA_PRICE, emaConf: HERMES_1_BTC_EMA_CONF, emaExpo: HERMES_1_BTC_EMA_EXPO, emaPublishTime: HERMES_1_BTC_EMA_PUBLISH_TIME },
+    ];
+
+    for (let i = 0; i < expectedFeeds.length; i++) {
+      const feedCs = currentCell.beginParse();
+      const priceId = "0x" + feedCs.loadUintBig(256).toString(16).padStart(64, "0");
+      expect(priceId).toBe(expectedFeeds[i].id);
+
+      const priceFeedCell = feedCs.loadRef();
+      const priceFeedSlice = priceFeedCell.beginParse();
+
+      const currentPriceCell = priceFeedSlice.loadRef();
+      const currentPrice = currentPriceCell.beginParse();
+      expect(currentPrice.loadInt(64)).toBe(expectedFeeds[i].price);
+      expect(currentPrice.loadUint(64)).toBe(expectedFeeds[i].conf);
+      expect(currentPrice.loadInt(32)).toBe(expectedFeeds[i].expo);
+      expect(currentPrice.loadUint(64)).toBe(expectedFeeds[i].publishTime);
+
+      const emaPriceCell = priceFeedSlice.loadRef();
+      const emaPrice = emaPriceCell.beginParse();
+      expect(emaPrice.loadInt(64)).toBe(expectedFeeds[i].emaPrice);
+      expect(emaPrice.loadUint(64)).toBe(expectedFeeds[i].emaConf);
+      expect(emaPrice.loadInt(32)).toBe(expectedFeeds[i].emaExpo);
+      expect(emaPrice.loadUint(64)).toBe(expectedFeeds[i].emaPublishTime);
+
+      if (i < expectedFeeds.length - 1) {
+        currentCell = feedCs.loadRef();
+      } else {
+        expect(feedCs.remainingRefs).toBe(0);
+      }
+    }
+
+    const senderAddress = cs.loadAddress();
+    expect(senderAddress?.toString()).toBe(
+      deployer.getSender().address.toString(),
+    );
+
+    const customPayloadCell = cs.loadRef();
+    const customPayloadSlice = customPayloadCell.beginParse();
+    const receivedPayload = Buffer.from(
+      customPayloadSlice.loadBuffer(CUSTOM_PAYLOAD.length),
+    );
+    expect(receivedPayload.toString("hex")).toBe(
+      CUSTOM_PAYLOAD.toString("hex"),
+    );
+  });
+
+  it("should successfully parse price feed updates with 7 price feeds", async () => {
+    await deployContract();
+    await updateGuardianSets(pythTest, deployer);
+
+    const sentValue = toNano("15");
+    const result = await pythTest.sendParsePriceFeedUpdates(
+      deployer.getSender(),
+      Buffer.from(HERMES_UPDATE_7_PRICE_FEEDS, "hex"),
+      sentValue,
+      [
+        BTC_PRICE_FEED_ID,
+        ETH_PRICE_FEED_ID,
+        DOGE_PRICE_FEED_ID,
+        SOL_USD_PRICE_FEED_ID,
+        PYTH_USD_PRICE_FEED_ID,
+        ARBITRUM_USD_PRICE_FEED_ID,
+        TON_USD_PRICE_FEED_ID,
+      ],
+      HERMES_7_ETH_PUBLISH_TIME,
+      HERMES_7_ETH_PUBLISH_TIME,
+      mockDeployer.address,
+      CUSTOM_PAYLOAD,
+    );
+
+    expect(result.transactions).toHaveTransaction({
+      from: deployer.address,
+      to: pythTest.address,
+      success: true,
+      outMessagesCount: 1,
+    });
+
+    expect(result.transactions).toHaveTransaction({
+      from: pythTest.address,
+      to: deployer.address,
+      success: true,
+    });
+
+    const outMessage = result.transactions[1].outMessages.values()[0];
+
+    expect(
+      (outMessage.info as CommonMessageInfoInternal).value.coins,
+    ).toBeGreaterThan(0);
+
+    const cs = outMessage.body.beginParse();
+
+    console.log(outMessage.body);
+
+    const op = cs.loadUint(32);
+    expect(op).toBe(5);
+
+    const numPriceFeeds = cs.loadUint(8);
+    expect(numPriceFeeds).toBe(10);
+
+    const priceFeedsCell = cs.loadRef();
+    let currentCell = priceFeedsCell;
+
+    const expectedFeeds = [
+      { id: BTC_PRICE_FEED_ID, price: HERMES_7_BTC_PRICE, conf: HERMES_7_BTC_CONF, expo: HERMES_7_BTC_EXPO, publishTime: HERMES_7_BTC_PUBLISH_TIME, emaPrice: HERMES_7_BTC_EMA_PRICE, emaConf: HERMES_7_BTC_EMA_CONF, emaExpo: HERMES_7_BTC_EMA_EXPO, emaPublishTime: HERMES_7_BTC_EMA_PUBLISH_TIME },
+      { id: ETH_PRICE_FEED_ID, price: HERMES_7_ETH_PRICE, conf: HERMES_7_ETH_CONF, expo: HERMES_7_ETH_EXPO, publishTime: HERMES_7_ETH_PUBLISH_TIME, emaPrice: HERMES_7_ETH_EMA_PRICE, emaConf: HERMES_7_ETH_EMA_CONF, emaExpo: HERMES_7_ETH_EMA_EXPO, emaPublishTime: HERMES_7_ETH_EMA_PUBLISH_TIME },
+      { id: DOGE_PRICE_FEED_ID, price: HERMES_7_DOGE_PRICE, conf: HERMES_7_DOGE_CONF, expo: HERMES_7_DOGE_EXPO, publishTime: HERMES_7_DOGE_PUBLISH_TIME, emaPrice: HERMES_7_DOGE_EMA_PRICE, emaConf: HERMES_7_DOGE_EMA_CONF, emaExpo: HERMES_7_DOGE_EMA_EXPO, emaPublishTime: HERMES_7_DOGE_EMA_PUBLISH_TIME },
+      { id: SOL_USD_PRICE_FEED_ID, price: HERMES_7_SOL_PRICE, conf: HERMES_7_SOL_CONF, expo: HERMES_7_SOL_EXPO, publishTime: HERMES_7_SOL_PUBLISH_TIME, emaPrice: HERMES_7_SOL_EMA_PRICE, emaConf: HERMES_7_SOL_EMA_CONF, emaExpo: HERMES_7_SOL_EMA_EXPO, emaPublishTime: HERMES_7_SOL_EMA_PUBLISH_TIME },
+      { id: PYTH_USD_PRICE_FEED_ID, price: HERMES_7_PYTH_PRICE, conf: HERMES_7_PYTH_CONF, expo: HERMES_7_PYTH_EXPO, publishTime: HERMES_7_PYTH_PUBLISH_TIME, emaPrice: HERMES_7_PYTH_EMA_PRICE, emaConf: HERMES_7_PYTH_EMA_CONF, emaExpo: HERMES_7_PYTH_EMA_EXPO, emaPublishTime: HERMES_7_PYTH_EMA_PUBLISH_TIME },
+      { id: ARBITRUM_USD_PRICE_FEED_ID, price: HERMES_7_ARB_PRICE, conf: HERMES_7_ARB_CONF, expo: HERMES_7_ARB_EXPO, publishTime: HERMES_7_ARB_PUBLISH_TIME, emaPrice: HERMES_7_ARB_EMA_PRICE, emaConf: HERMES_7_ARB_EMA_CONF, emaExpo: HERMES_7_ARB_EMA_EXPO, emaPublishTime: HERMES_7_ARB_EMA_PUBLISH_TIME },
+      { id: TON_USD_PRICE_FEED_ID, price: HERMES_7_TON_PRICE, conf: HERMES_7_TON_CONF, expo: HERMES_7_TON_EXPO, publishTime: HERMES_7_TON_PUBLISH_TIME, emaPrice: HERMES_7_TON_EMA_PRICE, emaConf: HERMES_7_TON_EMA_CONF, emaExpo: HERMES_7_TON_EMA_EXPO, emaPublishTime: HERMES_7_TON_EMA_PUBLISH_TIME },
+    ];
+
+    for (let i = 0; i < expectedFeeds.length; i++) {
+      const feedCs = currentCell.beginParse();
+      const priceId = "0x" + feedCs.loadUintBig(256).toString(16).padStart(64, "0");
+      expect(priceId).toBe(expectedFeeds[i].id);
+
+      const priceFeedCell = feedCs.loadRef();
+      const priceFeedSlice = priceFeedCell.beginParse();
+
+      const currentPriceCell = priceFeedSlice.loadRef();
+      const currentPrice = currentPriceCell.beginParse();
+      expect(currentPrice.loadInt(64)).toBe(expectedFeeds[i].price);
+      expect(currentPrice.loadUint(64)).toBe(expectedFeeds[i].conf);
+      expect(currentPrice.loadInt(32)).toBe(expectedFeeds[i].expo);
+      expect(currentPrice.loadUint(64)).toBe(expectedFeeds[i].publishTime);
+
+      const emaPriceCell = priceFeedSlice.loadRef();
+      const emaPrice = emaPriceCell.beginParse();
+      expect(emaPrice.loadInt(64)).toBe(expectedFeeds[i].emaPrice);
+      expect(emaPrice.loadUint(64)).toBe(expectedFeeds[i].emaConf);
+      expect(emaPrice.loadInt(32)).toBe(expectedFeeds[i].emaExpo);
+      expect(emaPrice.loadUint(64)).toBe(expectedFeeds[i].emaPublishTime);
+
+      if (i < expectedFeeds.length - 1) {
+        currentCell = feedCs.loadRef();
+      } else {
+        expect(feedCs.remainingRefs).toBe(0);
+      }
+    }
+
+    const senderAddress = cs.loadAddress();
+    expect(senderAddress?.toString()).toBe(
+      deployer.getSender().address.toString(),
+    );
+
+    const customPayloadCell = cs.loadRef();
+    const customPayloadSlice = customPayloadCell.beginParse();
+    const receivedPayload = Buffer.from(
+      customPayloadSlice.loadBuffer(CUSTOM_PAYLOAD.length),
+    );
+    expect(receivedPayload.toString("hex")).toBe(
+      CUSTOM_PAYLOAD.toString("hex"),
+    );
+  });
+
+
+  it("should successfully parse price feed updates with 10 price feeds", async () => {
+    await deployContract();
+    await updateGuardianSets(pythTest, deployer);
+
+    const sentValue = toNano("15");
     const result = await pythTest.sendParsePriceFeedUpdates(
       deployer.getSender(),
       Buffer.from(HERMES_UPDATE_10_PRICE_FEEDS, "hex"),
@@ -1627,13 +1991,12 @@ describe("PythTest", () => {
         ADBE_PRICE_FEED_ID,
         AMZN_PRICE_FEED_ID,
       ],
-      HERMES_BTC_PUBLISH_TIME,
-      HERMES_BTC_PUBLISH_TIME,
-      deployer.address,
+      HERMES_10_ETH_PUBLISH_TIME,
+      HERMES_10_ETH_PUBLISH_TIME,
+      mockDeployer.address,
       CUSTOM_PAYLOAD,
     );
 
-    // Verify transaction success and message count
     expect(result.transactions).toHaveTransaction({
       from: deployer.address,
       to: pythTest.address,
@@ -1641,40 +2004,77 @@ describe("PythTest", () => {
       outMessagesCount: 1,
     });
 
-    // Verify message success to target address
     expect(result.transactions).toHaveTransaction({
       from: pythTest.address,
-      to: mockDeployer.address,
+      to: deployer.address,
       success: true,
     });
 
-    // Get the output message
     const outMessage = result.transactions[1].outMessages.values()[0];
 
-    // Verify excess value is returned
     expect(
       (outMessage.info as CommonMessageInfoInternal).value.coins,
     ).toBeGreaterThan(0);
 
     const cs = outMessage.body.beginParse();
 
-    // Verify message header
     const op = cs.loadUint(32);
-    expect(op).toBe(5); // OP_PARSE_PRICE_FEED_UPDATES
+    expect(op).toBe(5);
 
-    // Verify number of price feeds
     const numPriceFeeds = cs.loadUint(8);
-    expect(numPriceFeeds).toBe(10); // We expect BTC and ETH price feeds
+    expect(numPriceFeeds).toBe(10);
 
-    cs.loadRef(); // Skip price feeds
+    const priceFeedsCell = cs.loadRef();
+    let currentCell = priceFeedsCell;
+
+    const expectedFeeds = [
+      { id: BTC_PRICE_FEED_ID, price: HERMES_10_BTC_PRICE, conf: HERMES_10_BTC_CONF, expo: HERMES_10_BTC_EXPO, publishTime: HERMES_10_BTC_PUBLISH_TIME, emaPrice: HERMES_10_BTC_EMA_PRICE, emaConf: HERMES_10_BTC_EMA_CONF, emaExpo: HERMES_10_BTC_EMA_EXPO, emaPublishTime: HERMES_10_BTC_EMA_PUBLISH_TIME },
+      { id: ETH_PRICE_FEED_ID, price: HERMES_10_ETH_PRICE, conf: HERMES_10_ETH_CONF, expo: HERMES_10_ETH_EXPO, publishTime: HERMES_10_ETH_PUBLISH_TIME, emaPrice: HERMES_10_ETH_EMA_PRICE, emaConf: HERMES_10_ETH_EMA_CONF, emaExpo: HERMES_10_ETH_EMA_EXPO, emaPublishTime: HERMES_10_ETH_EMA_PUBLISH_TIME },
+      { id: DOGE_PRICE_FEED_ID, price: HERMES_10_DOGE_PRICE, conf: HERMES_10_DOGE_CONF, expo: HERMES_10_DOGE_EXPO, publishTime: HERMES_10_DOGE_PUBLISH_TIME, emaPrice: HERMES_10_DOGE_EMA_PRICE, emaConf: HERMES_10_DOGE_EMA_CONF, emaExpo: HERMES_10_DOGE_EMA_EXPO, emaPublishTime: HERMES_10_DOGE_EMA_PUBLISH_TIME },
+      { id: SOL_USD_PRICE_FEED_ID, price: HERMES_10_SOL_PRICE, conf: HERMES_10_SOL_CONF, expo: HERMES_10_SOL_EXPO, publishTime: HERMES_10_SOL_PUBLISH_TIME, emaPrice: HERMES_10_SOL_EMA_PRICE, emaConf: HERMES_10_SOL_EMA_CONF, emaExpo: HERMES_10_SOL_EMA_EXPO, emaPublishTime: HERMES_10_SOL_EMA_PUBLISH_TIME },
+      { id: PYTH_USD_PRICE_FEED_ID, price: HERMES_10_PYTH_PRICE, conf: HERMES_10_PYTH_CONF, expo: HERMES_10_PYTH_EXPO, publishTime: HERMES_10_PYTH_PUBLISH_TIME, emaPrice: HERMES_10_PYTH_EMA_PRICE, emaConf: HERMES_10_PYTH_EMA_CONF, emaExpo: HERMES_10_PYTH_EMA_EXPO, emaPublishTime: HERMES_10_PYTH_EMA_PUBLISH_TIME },
+      { id: ARBITRUM_USD_PRICE_FEED_ID, price: HERMES_10_ARB_PRICE, conf: HERMES_10_ARB_CONF, expo: HERMES_10_ARB_EXPO, publishTime: HERMES_10_ARB_PUBLISH_TIME, emaPrice: HERMES_10_ARB_EMA_PRICE, emaConf: HERMES_10_ARB_EMA_CONF, emaExpo: HERMES_10_ARB_EMA_EXPO, emaPublishTime: HERMES_10_ARB_EMA_PUBLISH_TIME },
+      { id: TON_USD_PRICE_FEED_ID, price: HERMES_10_TON_PRICE, conf: HERMES_10_TON_CONF, expo: HERMES_10_TON_EXPO, publishTime: HERMES_10_TON_PUBLISH_TIME, emaPrice: HERMES_10_TON_EMA_PRICE, emaConf: HERMES_10_TON_EMA_CONF, emaExpo: HERMES_10_TON_EMA_EXPO, emaPublishTime: HERMES_10_TON_EMA_PUBLISH_TIME },
+      { id: AAPL_PRICE_FEED_ID, price: HERMES_10_AAPL_PRICE, conf: HERMES_10_AAPL_CONF, expo: HERMES_10_AAPL_EXPO, publishTime: HERMES_10_AAPL_PUBLISH_TIME, emaPrice: HERMES_10_AAPL_EMA_PRICE, emaConf: HERMES_10_AAPL_EMA_CONF, emaExpo: HERMES_10_AAPL_EMA_EXPO, emaPublishTime: HERMES_10_AAPL_EMA_PUBLISH_TIME },
+      { id: ABNB_PRICE_FEED_ID, price: HERMES_10_ABNB_PRICE, conf: HERMES_10_ABNB_CONF, expo: HERMES_10_ABNB_EXPO, publishTime: HERMES_10_ABNB_PUBLISH_TIME, emaPrice: HERMES_10_ABNB_EMA_PRICE, emaConf: HERMES_10_ABNB_EMA_CONF, emaExpo: HERMES_10_ABNB_EMA_EXPO, emaPublishTime: HERMES_10_ABNB_EMA_PUBLISH_TIME },
+      { id: ADBE_PRICE_FEED_ID, price: HERMES_10_ADBE_PRICE, conf: HERMES_10_ADBE_CONF, expo: HERMES_10_ADBE_EXPO, publishTime: HERMES_10_ADBE_PUBLISH_TIME, emaPrice: HERMES_10_ADBE_EMA_PRICE, emaConf: HERMES_10_ADBE_EMA_CONF, emaExpo: HERMES_10_ADBE_EMA_EXPO, emaPublishTime: HERMES_10_ADBE_EMA_PUBLISH_TIME },
+      { id: AMZN_PRICE_FEED_ID, price: HERMES_10_AMZN_PRICE, conf: HERMES_10_AMZN_CONF, expo: HERMES_10_AMZN_EXPO, publishTime: HERMES_10_AMZN_PUBLISH_TIME, emaPrice: HERMES_10_AMZN_EMA_PRICE, emaConf: HERMES_10_AMZN_EMA_CONF, emaExpo: HERMES_10_AMZN_EMA_EXPO, emaPublishTime: HERMES_10_AMZN_EMA_PUBLISH_TIME },
+    ];
+
+    for (let i = 0; i < expectedFeeds.length; i++) {
+      const feedCs = currentCell.beginParse();
+      const priceId = "0x" + feedCs.loadUintBig(256).toString(16).padStart(64, "0");
+      expect(priceId).toBe(expectedFeeds[i].id);
+
+      const priceFeedCell = feedCs.loadRef();
+      const priceFeedSlice = priceFeedCell.beginParse();
+
+      const currentPriceCell = priceFeedSlice.loadRef();
+      const currentPrice = currentPriceCell.beginParse();
+      expect(currentPrice.loadInt(64)).toBe(expectedFeeds[i].price);
+      expect(currentPrice.loadUint(64)).toBe(expectedFeeds[i].conf);
+      expect(currentPrice.loadInt(32)).toBe(expectedFeeds[i].expo);
+      expect(currentPrice.loadUint(64)).toBe(expectedFeeds[i].publishTime);
+
+      const emaPriceCell = priceFeedSlice.loadRef();
+      const emaPrice = emaPriceCell.beginParse();
+      expect(emaPrice.loadInt(64)).toBe(expectedFeeds[i].emaPrice);
+      expect(emaPrice.loadUint(64)).toBe(expectedFeeds[i].emaConf);
+      expect(emaPrice.loadInt(32)).toBe(expectedFeeds[i].emaExpo);
+      expect(emaPrice.loadUint(64)).toBe(expectedFeeds[i].emaPublishTime);
+
+      if (i < expectedFeeds.length - 1) {
+        currentCell = feedCs.loadRef();
+      } else {
+        expect(feedCs.remainingRefs).toBe(0);
+      }
+    }
 
-    // Verify sender address
     const senderAddress = cs.loadAddress();
     expect(senderAddress?.toString()).toBe(
       deployer.getSender().address.toString(),
     );
 
-    // Verify custom payload
     const customPayloadCell = cs.loadRef();
     const customPayloadSlice = customPayloadCell.beginParse();
     const receivedPayload = Buffer.from(

Filskillnaden har hållts tillbaka eftersom den är för stor
+ 0 - 0
target_chains/ton/contracts/tests/utils/pyth.ts


Vissa filer visades inte eftersom för många filer har ändrats