@pythnetwork/pyth-ton-js 0.2.0 → 0.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/index.cjs +303 -0
- package/{lib → dist/cjs}/index.d.ts +9 -9
- package/dist/cjs/package.json +1 -0
- package/dist/esm/index.d.ts +69 -0
- package/{lib/index.js → dist/esm/index.mjs} +82 -74
- package/dist/esm/package.json +1 -0
- package/package.json +28 -11
- package/lib/index.d.ts.map +0 -1
|
@@ -0,0 +1,303 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-non-null-assertion */ "use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", {
|
|
3
|
+
value: true
|
|
4
|
+
});
|
|
5
|
+
function _export(target, all) {
|
|
6
|
+
for(var name in all)Object.defineProperty(target, name, {
|
|
7
|
+
enumerable: true,
|
|
8
|
+
get: Object.getOwnPropertyDescriptor(all, name).get
|
|
9
|
+
});
|
|
10
|
+
}
|
|
11
|
+
_export(exports, {
|
|
12
|
+
get GAS_PRICE_FACTOR () {
|
|
13
|
+
return GAS_PRICE_FACTOR;
|
|
14
|
+
},
|
|
15
|
+
get PYTH_CONTRACT_ADDRESS_MAINNET () {
|
|
16
|
+
return PYTH_CONTRACT_ADDRESS_MAINNET;
|
|
17
|
+
},
|
|
18
|
+
get PYTH_CONTRACT_ADDRESS_TESTNET () {
|
|
19
|
+
return PYTH_CONTRACT_ADDRESS_TESTNET;
|
|
20
|
+
},
|
|
21
|
+
get PythContract () {
|
|
22
|
+
return PythContract;
|
|
23
|
+
},
|
|
24
|
+
get UPDATE_PRICE_FEEDS_BASE_GAS () {
|
|
25
|
+
return UPDATE_PRICE_FEEDS_BASE_GAS;
|
|
26
|
+
},
|
|
27
|
+
get UPDATE_PRICE_FEEDS_PER_UPDATE_GAS () {
|
|
28
|
+
return UPDATE_PRICE_FEEDS_PER_UPDATE_GAS;
|
|
29
|
+
},
|
|
30
|
+
get calculateUpdatePriceFeedsFee () {
|
|
31
|
+
return calculateUpdatePriceFeedsFee;
|
|
32
|
+
},
|
|
33
|
+
get createCellChain () {
|
|
34
|
+
return createCellChain;
|
|
35
|
+
},
|
|
36
|
+
get parseDataSource () {
|
|
37
|
+
return parseDataSource;
|
|
38
|
+
},
|
|
39
|
+
get parseDataSources () {
|
|
40
|
+
return parseDataSources;
|
|
41
|
+
},
|
|
42
|
+
get parseGuardianSetKeys () {
|
|
43
|
+
return parseGuardianSetKeys;
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
const _core = require("@ton/core");
|
|
47
|
+
const PYTH_CONTRACT_ADDRESS_MAINNET = "EQA5NPyjfZztDm8jcTBwTAU9NGsgJEkw19z61yecX0TlseSB";
|
|
48
|
+
const PYTH_CONTRACT_ADDRESS_TESTNET = "EQBeydTZBuv4nkqN8jkScqZbWcyx7TsmOtxsEDbFann0IRtD";
|
|
49
|
+
const UPDATE_PRICE_FEEDS_BASE_GAS = 300_000n;
|
|
50
|
+
const UPDATE_PRICE_FEEDS_PER_UPDATE_GAS = 90_000n;
|
|
51
|
+
const GAS_PRICE_FACTOR = 400n;
|
|
52
|
+
class PythContract {
|
|
53
|
+
address;
|
|
54
|
+
init;
|
|
55
|
+
constructor(address, init){
|
|
56
|
+
this.address = address;
|
|
57
|
+
this.init = init;
|
|
58
|
+
}
|
|
59
|
+
static createFromAddress(address) {
|
|
60
|
+
return new PythContract(address);
|
|
61
|
+
}
|
|
62
|
+
async getCurrentGuardianSetIndex(provider) {
|
|
63
|
+
const result = await provider.get("get_current_guardian_set_index", []);
|
|
64
|
+
return result.stack.readNumber();
|
|
65
|
+
}
|
|
66
|
+
async sendUpdateGuardianSet(provider, via, vm) {
|
|
67
|
+
const messageBody = (0, _core.beginCell)().storeUint(1, 32) // OP_UPDATE_GUARDIAN_SET
|
|
68
|
+
.storeRef(createCellChain(vm)).endCell();
|
|
69
|
+
await provider.internal(via, {
|
|
70
|
+
value: (0, _core.toNano)("0.1"),
|
|
71
|
+
sendMode: _core.SendMode.PAY_GAS_SEPARATELY,
|
|
72
|
+
body: messageBody
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
async sendUpdatePriceFeeds(provider, via, updateData, updateFee) {
|
|
76
|
+
const messageBody = (0, _core.beginCell)().storeUint(2, 32) // OP_UPDATE_PRICE_FEEDS
|
|
77
|
+
.storeRef(createCellChain(updateData)).endCell();
|
|
78
|
+
await provider.internal(via, {
|
|
79
|
+
value: updateFee,
|
|
80
|
+
sendMode: _core.SendMode.PAY_GAS_SEPARATELY,
|
|
81
|
+
body: messageBody
|
|
82
|
+
});
|
|
83
|
+
}
|
|
84
|
+
async sendExecuteGovernanceAction(provider, via, governanceAction) {
|
|
85
|
+
const messageBody = (0, _core.beginCell)().storeUint(3, 32) // OP_EXECUTE_GOVERNANCE_ACTION
|
|
86
|
+
.storeRef(createCellChain(governanceAction)).endCell();
|
|
87
|
+
await provider.internal(via, {
|
|
88
|
+
value: (0, _core.toNano)("0.1"),
|
|
89
|
+
sendMode: _core.SendMode.PAY_GAS_SEPARATELY,
|
|
90
|
+
body: messageBody
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
async sendUpgradeContract(provider, via, newCode) {
|
|
94
|
+
const messageBody = (0, _core.beginCell)().storeUint(4, 32) // OP_UPGRADE_CONTRACT
|
|
95
|
+
.storeRef(newCode).endCell();
|
|
96
|
+
await provider.internal(via, {
|
|
97
|
+
value: (0, _core.toNano)("0.1"),
|
|
98
|
+
sendMode: _core.SendMode.PAY_GAS_SEPARATELY,
|
|
99
|
+
body: messageBody
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
async getPriceUnsafe(provider, priceFeedId) {
|
|
103
|
+
const result = await provider.get("get_price_unsafe", [
|
|
104
|
+
{
|
|
105
|
+
type: "int",
|
|
106
|
+
value: BigInt(priceFeedId)
|
|
107
|
+
}
|
|
108
|
+
]);
|
|
109
|
+
const price = result.stack.readNumber();
|
|
110
|
+
const conf = result.stack.readNumber();
|
|
111
|
+
const expo = result.stack.readNumber();
|
|
112
|
+
const publishTime = result.stack.readNumber();
|
|
113
|
+
return {
|
|
114
|
+
price,
|
|
115
|
+
conf,
|
|
116
|
+
expo,
|
|
117
|
+
publishTime
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
async getPriceNoOlderThan(provider, timePeriod, priceFeedId) {
|
|
121
|
+
const result = await provider.get("get_price_no_older_than", [
|
|
122
|
+
{
|
|
123
|
+
type: "int",
|
|
124
|
+
value: BigInt(timePeriod)
|
|
125
|
+
},
|
|
126
|
+
{
|
|
127
|
+
type: "int",
|
|
128
|
+
value: BigInt(priceFeedId)
|
|
129
|
+
}
|
|
130
|
+
]);
|
|
131
|
+
const price = result.stack.readNumber();
|
|
132
|
+
const conf = result.stack.readNumber();
|
|
133
|
+
const expo = result.stack.readNumber();
|
|
134
|
+
const publishTime = result.stack.readNumber();
|
|
135
|
+
return {
|
|
136
|
+
price,
|
|
137
|
+
conf,
|
|
138
|
+
expo,
|
|
139
|
+
publishTime
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
async getEmaPriceUnsafe(provider, priceFeedId) {
|
|
143
|
+
const result = await provider.get("get_ema_price_unsafe", [
|
|
144
|
+
{
|
|
145
|
+
type: "int",
|
|
146
|
+
value: BigInt(priceFeedId)
|
|
147
|
+
}
|
|
148
|
+
]);
|
|
149
|
+
const price = result.stack.readNumber();
|
|
150
|
+
const conf = result.stack.readNumber();
|
|
151
|
+
const expo = result.stack.readNumber();
|
|
152
|
+
const publishTime = result.stack.readNumber();
|
|
153
|
+
return {
|
|
154
|
+
price,
|
|
155
|
+
conf,
|
|
156
|
+
expo,
|
|
157
|
+
publishTime
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
async getEmaPriceNoOlderThan(provider, timePeriod, priceFeedId) {
|
|
161
|
+
const result = await provider.get("get_ema_price_no_older_than", [
|
|
162
|
+
{
|
|
163
|
+
type: "int",
|
|
164
|
+
value: BigInt(timePeriod)
|
|
165
|
+
},
|
|
166
|
+
{
|
|
167
|
+
type: "int",
|
|
168
|
+
value: BigInt(priceFeedId)
|
|
169
|
+
}
|
|
170
|
+
]);
|
|
171
|
+
const price = result.stack.readNumber();
|
|
172
|
+
const conf = result.stack.readNumber();
|
|
173
|
+
const expo = result.stack.readNumber();
|
|
174
|
+
const publishTime = result.stack.readNumber();
|
|
175
|
+
return {
|
|
176
|
+
price,
|
|
177
|
+
conf,
|
|
178
|
+
expo,
|
|
179
|
+
publishTime
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
async getUpdateFee(provider, vm) {
|
|
183
|
+
const result = await provider.get("get_update_fee", [
|
|
184
|
+
{
|
|
185
|
+
type: "slice",
|
|
186
|
+
cell: createCellChain(vm)
|
|
187
|
+
}
|
|
188
|
+
]);
|
|
189
|
+
return result.stack.readNumber();
|
|
190
|
+
}
|
|
191
|
+
async getSingleUpdateFee(provider) {
|
|
192
|
+
const result = await provider.get("get_single_update_fee", []);
|
|
193
|
+
return result.stack.readNumber();
|
|
194
|
+
}
|
|
195
|
+
async getLastExecutedGovernanceSequence(provider) {
|
|
196
|
+
const result = await provider.get("get_last_executed_governance_sequence", []);
|
|
197
|
+
return result.stack.readNumber();
|
|
198
|
+
}
|
|
199
|
+
async getChainId(provider) {
|
|
200
|
+
const result = await provider.get("get_chain_id", []);
|
|
201
|
+
return result.stack.readNumber();
|
|
202
|
+
}
|
|
203
|
+
async getDataSources(provider) {
|
|
204
|
+
const result = await provider.get("get_data_sources", []);
|
|
205
|
+
return parseDataSources(result.stack.readCell());
|
|
206
|
+
}
|
|
207
|
+
async getGovernanceDataSource(provider) {
|
|
208
|
+
const result = await provider.get("get_governance_data_source", []);
|
|
209
|
+
return parseDataSource(result.stack.readCell());
|
|
210
|
+
}
|
|
211
|
+
async getGuardianSet(provider, index) {
|
|
212
|
+
const result = await provider.get("get_guardian_set", [
|
|
213
|
+
{
|
|
214
|
+
type: "int",
|
|
215
|
+
value: BigInt(index)
|
|
216
|
+
}
|
|
217
|
+
]);
|
|
218
|
+
const expirationTime = result.stack.readNumber();
|
|
219
|
+
const keys = parseGuardianSetKeys(result.stack.readCell());
|
|
220
|
+
const keyCount = result.stack.readNumber();
|
|
221
|
+
return {
|
|
222
|
+
expirationTime,
|
|
223
|
+
keys,
|
|
224
|
+
keyCount
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
function createCellChain(buffer) {
|
|
229
|
+
const chunks = bufferToChunks(buffer, 127);
|
|
230
|
+
let lastCell;
|
|
231
|
+
// Iterate through chunks in reverse order
|
|
232
|
+
for(let i = chunks.length - 1; i >= 0; i--){
|
|
233
|
+
const chunk = chunks[i];
|
|
234
|
+
const cellBuilder = (0, _core.beginCell)();
|
|
235
|
+
const buffer = Buffer.from(chunk);
|
|
236
|
+
cellBuilder.storeBuffer(buffer);
|
|
237
|
+
if (lastCell) {
|
|
238
|
+
cellBuilder.storeRef(lastCell);
|
|
239
|
+
}
|
|
240
|
+
lastCell = cellBuilder.endCell();
|
|
241
|
+
}
|
|
242
|
+
// lastCell will be the root cell of our chain
|
|
243
|
+
if (!lastCell) {
|
|
244
|
+
throw new Error("Failed to create cell chain");
|
|
245
|
+
}
|
|
246
|
+
return lastCell;
|
|
247
|
+
}
|
|
248
|
+
function bufferToChunks(buff, chunkSizeBytes = 127) {
|
|
249
|
+
const chunks = [];
|
|
250
|
+
const uint8Array = new Uint8Array(buff.buffer, buff.byteOffset, buff.byteLength);
|
|
251
|
+
for(let offset = 0; offset < uint8Array.length; offset += chunkSizeBytes){
|
|
252
|
+
const remainingBytes = Math.min(chunkSizeBytes, uint8Array.length - offset);
|
|
253
|
+
const chunk = uint8Array.subarray(offset, offset + remainingBytes);
|
|
254
|
+
chunks.push(chunk);
|
|
255
|
+
}
|
|
256
|
+
return chunks;
|
|
257
|
+
}
|
|
258
|
+
function parseDataSources(cell) {
|
|
259
|
+
const dataSources = [];
|
|
260
|
+
const slice = cell.beginParse();
|
|
261
|
+
const dict = slice.loadDictDirect(_core.Dictionary.Keys.Uint(8), _core.Dictionary.Values.Cell());
|
|
262
|
+
for (const [, value] of dict){
|
|
263
|
+
const dataSource = parseDataSource(value);
|
|
264
|
+
if (dataSource) {
|
|
265
|
+
dataSources.push(dataSource);
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
return dataSources;
|
|
269
|
+
}
|
|
270
|
+
function parseDataSource(cell) {
|
|
271
|
+
const slice = cell.beginParse();
|
|
272
|
+
if (slice.remainingBits === 0) {
|
|
273
|
+
return;
|
|
274
|
+
}
|
|
275
|
+
const emitterChain = slice.loadUint(16);
|
|
276
|
+
const emitterAddress = slice.loadUintBig(256).toString(16).padStart(64, "0");
|
|
277
|
+
return {
|
|
278
|
+
emitterChain,
|
|
279
|
+
emitterAddress
|
|
280
|
+
};
|
|
281
|
+
}
|
|
282
|
+
function parseGuardianSetKeys(cell) {
|
|
283
|
+
const keys = [];
|
|
284
|
+
function parseCell(c) {
|
|
285
|
+
let slice = c.beginParse();
|
|
286
|
+
while(slice.remainingRefs > 0 || slice.remainingBits >= 160){
|
|
287
|
+
if (slice.remainingBits >= 160) {
|
|
288
|
+
const bitsToSkip = slice.remainingBits - 160;
|
|
289
|
+
slice = slice.skip(bitsToSkip);
|
|
290
|
+
const key = slice.loadBits(160);
|
|
291
|
+
keys.push("0x" + key.toString());
|
|
292
|
+
}
|
|
293
|
+
if (slice.remainingRefs > 0) {
|
|
294
|
+
parseCell(slice.loadRef());
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
parseCell(cell);
|
|
299
|
+
return keys;
|
|
300
|
+
}
|
|
301
|
+
function calculateUpdatePriceFeedsFee(numUpdates) {
|
|
302
|
+
return (UPDATE_PRICE_FEEDS_BASE_GAS + UPDATE_PRICE_FEEDS_PER_UPDATE_GAS * numUpdates) * GAS_PRICE_FACTOR;
|
|
303
|
+
}
|
|
@@ -1,14 +1,15 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
3
|
-
|
|
4
|
-
export declare const
|
|
1
|
+
import type { Contract, Sender } from "@ton/core";
|
|
2
|
+
import { Address, Cell } from "@ton/core";
|
|
3
|
+
import type { ContractProvider } from "@ton/ton";
|
|
4
|
+
export declare const PYTH_CONTRACT_ADDRESS_MAINNET = "EQA5NPyjfZztDm8jcTBwTAU9NGsgJEkw19z61yecX0TlseSB";
|
|
5
|
+
export declare const PYTH_CONTRACT_ADDRESS_TESTNET = "EQBeydTZBuv4nkqN8jkScqZbWcyx7TsmOtxsEDbFann0IRtD";
|
|
5
6
|
export declare const UPDATE_PRICE_FEEDS_BASE_GAS = 300000n;
|
|
6
7
|
export declare const UPDATE_PRICE_FEEDS_PER_UPDATE_GAS = 90000n;
|
|
7
8
|
export declare const GAS_PRICE_FACTOR = 400n;
|
|
8
|
-
export
|
|
9
|
+
export type DataSource = {
|
|
9
10
|
emitterChain: number;
|
|
10
11
|
emitterAddress: string;
|
|
11
|
-
}
|
|
12
|
+
};
|
|
12
13
|
export declare class PythContract implements Contract {
|
|
13
14
|
readonly address: Address;
|
|
14
15
|
readonly init?: {
|
|
@@ -54,7 +55,7 @@ export declare class PythContract implements Contract {
|
|
|
54
55
|
getLastExecutedGovernanceSequence(provider: ContractProvider): Promise<number>;
|
|
55
56
|
getChainId(provider: ContractProvider): Promise<number>;
|
|
56
57
|
getDataSources(provider: ContractProvider): Promise<DataSource[]>;
|
|
57
|
-
getGovernanceDataSource(provider: ContractProvider): Promise<DataSource | null>;
|
|
58
|
+
getGovernanceDataSource(provider: ContractProvider): Promise<DataSource | null | undefined>;
|
|
58
59
|
getGuardianSet(provider: ContractProvider, index: number): Promise<{
|
|
59
60
|
expirationTime: number;
|
|
60
61
|
keys: string[];
|
|
@@ -63,7 +64,6 @@ export declare class PythContract implements Contract {
|
|
|
63
64
|
}
|
|
64
65
|
export declare function createCellChain(buffer: Buffer): Cell;
|
|
65
66
|
export declare function parseDataSources(cell: Cell): DataSource[];
|
|
66
|
-
export declare function parseDataSource(cell: Cell): DataSource | null;
|
|
67
|
+
export declare function parseDataSource(cell: Cell): DataSource | null | undefined;
|
|
67
68
|
export declare function parseGuardianSetKeys(cell: Cell): string[];
|
|
68
69
|
export declare function calculateUpdatePriceFeedsFee(numUpdates: bigint): bigint;
|
|
69
|
-
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{ "type": "commonjs" }
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import type { Contract, Sender } from "@ton/core";
|
|
2
|
+
import { Address, Cell } from "@ton/core";
|
|
3
|
+
import type { ContractProvider } from "@ton/ton";
|
|
4
|
+
export declare const PYTH_CONTRACT_ADDRESS_MAINNET = "EQA5NPyjfZztDm8jcTBwTAU9NGsgJEkw19z61yecX0TlseSB";
|
|
5
|
+
export declare const PYTH_CONTRACT_ADDRESS_TESTNET = "EQBeydTZBuv4nkqN8jkScqZbWcyx7TsmOtxsEDbFann0IRtD";
|
|
6
|
+
export declare const UPDATE_PRICE_FEEDS_BASE_GAS = 300000n;
|
|
7
|
+
export declare const UPDATE_PRICE_FEEDS_PER_UPDATE_GAS = 90000n;
|
|
8
|
+
export declare const GAS_PRICE_FACTOR = 400n;
|
|
9
|
+
export type DataSource = {
|
|
10
|
+
emitterChain: number;
|
|
11
|
+
emitterAddress: string;
|
|
12
|
+
};
|
|
13
|
+
export declare class PythContract implements Contract {
|
|
14
|
+
readonly address: Address;
|
|
15
|
+
readonly init?: {
|
|
16
|
+
code: Cell;
|
|
17
|
+
data: Cell;
|
|
18
|
+
} | undefined;
|
|
19
|
+
constructor(address: Address, init?: {
|
|
20
|
+
code: Cell;
|
|
21
|
+
data: Cell;
|
|
22
|
+
} | undefined);
|
|
23
|
+
static createFromAddress(address: Address): PythContract;
|
|
24
|
+
getCurrentGuardianSetIndex(provider: ContractProvider): Promise<number>;
|
|
25
|
+
sendUpdateGuardianSet(provider: ContractProvider, via: Sender, vm: Buffer): Promise<void>;
|
|
26
|
+
sendUpdatePriceFeeds(provider: ContractProvider, via: Sender, updateData: Buffer, updateFee: bigint): Promise<void>;
|
|
27
|
+
sendExecuteGovernanceAction(provider: ContractProvider, via: Sender, governanceAction: Buffer): Promise<void>;
|
|
28
|
+
sendUpgradeContract(provider: ContractProvider, via: Sender, newCode: Cell): Promise<void>;
|
|
29
|
+
getPriceUnsafe(provider: ContractProvider, priceFeedId: string): Promise<{
|
|
30
|
+
price: number;
|
|
31
|
+
conf: number;
|
|
32
|
+
expo: number;
|
|
33
|
+
publishTime: number;
|
|
34
|
+
}>;
|
|
35
|
+
getPriceNoOlderThan(provider: ContractProvider, timePeriod: number, priceFeedId: string): Promise<{
|
|
36
|
+
price: number;
|
|
37
|
+
conf: number;
|
|
38
|
+
expo: number;
|
|
39
|
+
publishTime: number;
|
|
40
|
+
}>;
|
|
41
|
+
getEmaPriceUnsafe(provider: ContractProvider, priceFeedId: string): Promise<{
|
|
42
|
+
price: number;
|
|
43
|
+
conf: number;
|
|
44
|
+
expo: number;
|
|
45
|
+
publishTime: number;
|
|
46
|
+
}>;
|
|
47
|
+
getEmaPriceNoOlderThan(provider: ContractProvider, timePeriod: number, priceFeedId: string): Promise<{
|
|
48
|
+
price: number;
|
|
49
|
+
conf: number;
|
|
50
|
+
expo: number;
|
|
51
|
+
publishTime: number;
|
|
52
|
+
}>;
|
|
53
|
+
getUpdateFee(provider: ContractProvider, vm: Buffer): Promise<number>;
|
|
54
|
+
getSingleUpdateFee(provider: ContractProvider): Promise<number>;
|
|
55
|
+
getLastExecutedGovernanceSequence(provider: ContractProvider): Promise<number>;
|
|
56
|
+
getChainId(provider: ContractProvider): Promise<number>;
|
|
57
|
+
getDataSources(provider: ContractProvider): Promise<DataSource[]>;
|
|
58
|
+
getGovernanceDataSource(provider: ContractProvider): Promise<DataSource | null | undefined>;
|
|
59
|
+
getGuardianSet(provider: ContractProvider, index: number): Promise<{
|
|
60
|
+
expirationTime: number;
|
|
61
|
+
keys: string[];
|
|
62
|
+
keyCount: number;
|
|
63
|
+
}>;
|
|
64
|
+
}
|
|
65
|
+
export declare function createCellChain(buffer: Buffer): Cell;
|
|
66
|
+
export declare function parseDataSources(cell: Cell): DataSource[];
|
|
67
|
+
export declare function parseDataSource(cell: Cell): DataSource | null | undefined;
|
|
68
|
+
export declare function parseGuardianSetKeys(cell: Cell): string[];
|
|
69
|
+
export declare function calculateUpdatePriceFeedsFee(numUpdates: bigint): bigint;
|
|
@@ -1,23 +1,15 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
exports.createCellChain = createCellChain;
|
|
5
|
-
exports.parseDataSources = parseDataSources;
|
|
6
|
-
exports.parseDataSource = parseDataSource;
|
|
7
|
-
exports.parseGuardianSetKeys = parseGuardianSetKeys;
|
|
8
|
-
exports.calculateUpdatePriceFeedsFee = calculateUpdatePriceFeedsFee;
|
|
9
|
-
const core_1 = require("@ton/core");
|
|
10
|
-
exports.PYTH_CONTRACT_ADDRESS_MAINNET = "EQBgtfuGIzWLiOzpZO48_psYvco4xRtkAbdbmTwy0_o95LtZ";
|
|
11
|
-
exports.PYTH_CONTRACT_ADDRESS_TESTNET = "EQB4ZnrI5qsP_IUJgVJNwEGKLzZWsQOFhiaqDbD7pTt_f9oU";
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-non-null-assertion */ import { beginCell, Dictionary, SendMode, toNano } from "@ton/core";
|
|
2
|
+
export const PYTH_CONTRACT_ADDRESS_MAINNET = "EQA5NPyjfZztDm8jcTBwTAU9NGsgJEkw19z61yecX0TlseSB";
|
|
3
|
+
export const PYTH_CONTRACT_ADDRESS_TESTNET = "EQBeydTZBuv4nkqN8jkScqZbWcyx7TsmOtxsEDbFann0IRtD";
|
|
12
4
|
// This is defined in target_chains/ton/contracts/common/gas.fc
|
|
13
|
-
|
|
14
|
-
|
|
5
|
+
export const UPDATE_PRICE_FEEDS_BASE_GAS = 300_000n;
|
|
6
|
+
export const UPDATE_PRICE_FEEDS_PER_UPDATE_GAS = 90_000n;
|
|
15
7
|
// Current settings in basechain are as follows: 1 unit of gas costs 400 nanotons
|
|
16
|
-
|
|
17
|
-
class PythContract {
|
|
8
|
+
export const GAS_PRICE_FACTOR = 400n;
|
|
9
|
+
export class PythContract {
|
|
18
10
|
address;
|
|
19
11
|
init;
|
|
20
|
-
constructor(address, init)
|
|
12
|
+
constructor(address, init){
|
|
21
13
|
this.address = address;
|
|
22
14
|
this.init = init;
|
|
23
15
|
}
|
|
@@ -29,52 +21,47 @@ class PythContract {
|
|
|
29
21
|
return result.stack.readNumber();
|
|
30
22
|
}
|
|
31
23
|
async sendUpdateGuardianSet(provider, via, vm) {
|
|
32
|
-
const messageBody = (
|
|
33
|
-
|
|
34
|
-
.storeRef(createCellChain(vm))
|
|
35
|
-
.endCell();
|
|
24
|
+
const messageBody = beginCell().storeUint(1, 32) // OP_UPDATE_GUARDIAN_SET
|
|
25
|
+
.storeRef(createCellChain(vm)).endCell();
|
|
36
26
|
await provider.internal(via, {
|
|
37
|
-
value:
|
|
38
|
-
sendMode:
|
|
39
|
-
body: messageBody
|
|
27
|
+
value: toNano("0.1"),
|
|
28
|
+
sendMode: SendMode.PAY_GAS_SEPARATELY,
|
|
29
|
+
body: messageBody
|
|
40
30
|
});
|
|
41
31
|
}
|
|
42
32
|
async sendUpdatePriceFeeds(provider, via, updateData, updateFee) {
|
|
43
|
-
const messageBody = (
|
|
44
|
-
|
|
45
|
-
.storeRef(createCellChain(updateData))
|
|
46
|
-
.endCell();
|
|
33
|
+
const messageBody = beginCell().storeUint(2, 32) // OP_UPDATE_PRICE_FEEDS
|
|
34
|
+
.storeRef(createCellChain(updateData)).endCell();
|
|
47
35
|
await provider.internal(via, {
|
|
48
36
|
value: updateFee,
|
|
49
|
-
sendMode:
|
|
50
|
-
body: messageBody
|
|
37
|
+
sendMode: SendMode.PAY_GAS_SEPARATELY,
|
|
38
|
+
body: messageBody
|
|
51
39
|
});
|
|
52
40
|
}
|
|
53
41
|
async sendExecuteGovernanceAction(provider, via, governanceAction) {
|
|
54
|
-
const messageBody = (
|
|
55
|
-
|
|
56
|
-
.storeRef(createCellChain(governanceAction))
|
|
57
|
-
.endCell();
|
|
42
|
+
const messageBody = beginCell().storeUint(3, 32) // OP_EXECUTE_GOVERNANCE_ACTION
|
|
43
|
+
.storeRef(createCellChain(governanceAction)).endCell();
|
|
58
44
|
await provider.internal(via, {
|
|
59
|
-
value:
|
|
60
|
-
sendMode:
|
|
61
|
-
body: messageBody
|
|
45
|
+
value: toNano("0.1"),
|
|
46
|
+
sendMode: SendMode.PAY_GAS_SEPARATELY,
|
|
47
|
+
body: messageBody
|
|
62
48
|
});
|
|
63
49
|
}
|
|
64
50
|
async sendUpgradeContract(provider, via, newCode) {
|
|
65
|
-
const messageBody = (
|
|
66
|
-
|
|
67
|
-
.storeRef(newCode)
|
|
68
|
-
.endCell();
|
|
51
|
+
const messageBody = beginCell().storeUint(4, 32) // OP_UPGRADE_CONTRACT
|
|
52
|
+
.storeRef(newCode).endCell();
|
|
69
53
|
await provider.internal(via, {
|
|
70
|
-
value:
|
|
71
|
-
sendMode:
|
|
72
|
-
body: messageBody
|
|
54
|
+
value: toNano("0.1"),
|
|
55
|
+
sendMode: SendMode.PAY_GAS_SEPARATELY,
|
|
56
|
+
body: messageBody
|
|
73
57
|
});
|
|
74
58
|
}
|
|
75
59
|
async getPriceUnsafe(provider, priceFeedId) {
|
|
76
60
|
const result = await provider.get("get_price_unsafe", [
|
|
77
|
-
{
|
|
61
|
+
{
|
|
62
|
+
type: "int",
|
|
63
|
+
value: BigInt(priceFeedId)
|
|
64
|
+
}
|
|
78
65
|
]);
|
|
79
66
|
const price = result.stack.readNumber();
|
|
80
67
|
const conf = result.stack.readNumber();
|
|
@@ -84,13 +71,19 @@ class PythContract {
|
|
|
84
71
|
price,
|
|
85
72
|
conf,
|
|
86
73
|
expo,
|
|
87
|
-
publishTime
|
|
74
|
+
publishTime
|
|
88
75
|
};
|
|
89
76
|
}
|
|
90
77
|
async getPriceNoOlderThan(provider, timePeriod, priceFeedId) {
|
|
91
78
|
const result = await provider.get("get_price_no_older_than", [
|
|
92
|
-
{
|
|
93
|
-
|
|
79
|
+
{
|
|
80
|
+
type: "int",
|
|
81
|
+
value: BigInt(timePeriod)
|
|
82
|
+
},
|
|
83
|
+
{
|
|
84
|
+
type: "int",
|
|
85
|
+
value: BigInt(priceFeedId)
|
|
86
|
+
}
|
|
94
87
|
]);
|
|
95
88
|
const price = result.stack.readNumber();
|
|
96
89
|
const conf = result.stack.readNumber();
|
|
@@ -100,12 +93,15 @@ class PythContract {
|
|
|
100
93
|
price,
|
|
101
94
|
conf,
|
|
102
95
|
expo,
|
|
103
|
-
publishTime
|
|
96
|
+
publishTime
|
|
104
97
|
};
|
|
105
98
|
}
|
|
106
99
|
async getEmaPriceUnsafe(provider, priceFeedId) {
|
|
107
100
|
const result = await provider.get("get_ema_price_unsafe", [
|
|
108
|
-
{
|
|
101
|
+
{
|
|
102
|
+
type: "int",
|
|
103
|
+
value: BigInt(priceFeedId)
|
|
104
|
+
}
|
|
109
105
|
]);
|
|
110
106
|
const price = result.stack.readNumber();
|
|
111
107
|
const conf = result.stack.readNumber();
|
|
@@ -115,13 +111,19 @@ class PythContract {
|
|
|
115
111
|
price,
|
|
116
112
|
conf,
|
|
117
113
|
expo,
|
|
118
|
-
publishTime
|
|
114
|
+
publishTime
|
|
119
115
|
};
|
|
120
116
|
}
|
|
121
117
|
async getEmaPriceNoOlderThan(provider, timePeriod, priceFeedId) {
|
|
122
118
|
const result = await provider.get("get_ema_price_no_older_than", [
|
|
123
|
-
{
|
|
124
|
-
|
|
119
|
+
{
|
|
120
|
+
type: "int",
|
|
121
|
+
value: BigInt(timePeriod)
|
|
122
|
+
},
|
|
123
|
+
{
|
|
124
|
+
type: "int",
|
|
125
|
+
value: BigInt(priceFeedId)
|
|
126
|
+
}
|
|
125
127
|
]);
|
|
126
128
|
const price = result.stack.readNumber();
|
|
127
129
|
const conf = result.stack.readNumber();
|
|
@@ -131,12 +133,15 @@ class PythContract {
|
|
|
131
133
|
price,
|
|
132
134
|
conf,
|
|
133
135
|
expo,
|
|
134
|
-
publishTime
|
|
136
|
+
publishTime
|
|
135
137
|
};
|
|
136
138
|
}
|
|
137
139
|
async getUpdateFee(provider, vm) {
|
|
138
140
|
const result = await provider.get("get_update_fee", [
|
|
139
|
-
{
|
|
141
|
+
{
|
|
142
|
+
type: "slice",
|
|
143
|
+
cell: createCellChain(vm)
|
|
144
|
+
}
|
|
140
145
|
]);
|
|
141
146
|
return result.stack.readNumber();
|
|
142
147
|
}
|
|
@@ -162,7 +167,10 @@ class PythContract {
|
|
|
162
167
|
}
|
|
163
168
|
async getGuardianSet(provider, index) {
|
|
164
169
|
const result = await provider.get("get_guardian_set", [
|
|
165
|
-
{
|
|
170
|
+
{
|
|
171
|
+
type: "int",
|
|
172
|
+
value: BigInt(index)
|
|
173
|
+
}
|
|
166
174
|
]);
|
|
167
175
|
const expirationTime = result.stack.readNumber();
|
|
168
176
|
const keys = parseGuardianSetKeys(result.stack.readCell());
|
|
@@ -170,18 +178,17 @@ class PythContract {
|
|
|
170
178
|
return {
|
|
171
179
|
expirationTime,
|
|
172
180
|
keys,
|
|
173
|
-
keyCount
|
|
181
|
+
keyCount
|
|
174
182
|
};
|
|
175
183
|
}
|
|
176
184
|
}
|
|
177
|
-
|
|
178
|
-
function createCellChain(buffer) {
|
|
185
|
+
export function createCellChain(buffer) {
|
|
179
186
|
const chunks = bufferToChunks(buffer, 127);
|
|
180
|
-
let lastCell
|
|
187
|
+
let lastCell;
|
|
181
188
|
// Iterate through chunks in reverse order
|
|
182
|
-
for
|
|
189
|
+
for(let i = chunks.length - 1; i >= 0; i--){
|
|
183
190
|
const chunk = chunks[i];
|
|
184
|
-
const cellBuilder =
|
|
191
|
+
const cellBuilder = beginCell();
|
|
185
192
|
const buffer = Buffer.from(chunk);
|
|
186
193
|
cellBuilder.storeBuffer(buffer);
|
|
187
194
|
if (lastCell) {
|
|
@@ -198,18 +205,18 @@ function createCellChain(buffer) {
|
|
|
198
205
|
function bufferToChunks(buff, chunkSizeBytes = 127) {
|
|
199
206
|
const chunks = [];
|
|
200
207
|
const uint8Array = new Uint8Array(buff.buffer, buff.byteOffset, buff.byteLength);
|
|
201
|
-
for
|
|
208
|
+
for(let offset = 0; offset < uint8Array.length; offset += chunkSizeBytes){
|
|
202
209
|
const remainingBytes = Math.min(chunkSizeBytes, uint8Array.length - offset);
|
|
203
210
|
const chunk = uint8Array.subarray(offset, offset + remainingBytes);
|
|
204
211
|
chunks.push(chunk);
|
|
205
212
|
}
|
|
206
213
|
return chunks;
|
|
207
214
|
}
|
|
208
|
-
function parseDataSources(cell) {
|
|
215
|
+
export function parseDataSources(cell) {
|
|
209
216
|
const dataSources = [];
|
|
210
217
|
const slice = cell.beginParse();
|
|
211
|
-
const dict = slice.loadDictDirect(
|
|
212
|
-
for (const [, value] of dict)
|
|
218
|
+
const dict = slice.loadDictDirect(Dictionary.Keys.Uint(8), Dictionary.Values.Cell());
|
|
219
|
+
for (const [, value] of dict){
|
|
213
220
|
const dataSource = parseDataSource(value);
|
|
214
221
|
if (dataSource) {
|
|
215
222
|
dataSources.push(dataSource);
|
|
@@ -217,20 +224,23 @@ function parseDataSources(cell) {
|
|
|
217
224
|
}
|
|
218
225
|
return dataSources;
|
|
219
226
|
}
|
|
220
|
-
function parseDataSource(cell) {
|
|
227
|
+
export function parseDataSource(cell) {
|
|
221
228
|
const slice = cell.beginParse();
|
|
222
229
|
if (slice.remainingBits === 0) {
|
|
223
|
-
return
|
|
230
|
+
return;
|
|
224
231
|
}
|
|
225
232
|
const emitterChain = slice.loadUint(16);
|
|
226
233
|
const emitterAddress = slice.loadUintBig(256).toString(16).padStart(64, "0");
|
|
227
|
-
return {
|
|
234
|
+
return {
|
|
235
|
+
emitterChain,
|
|
236
|
+
emitterAddress
|
|
237
|
+
};
|
|
228
238
|
}
|
|
229
|
-
function parseGuardianSetKeys(cell) {
|
|
239
|
+
export function parseGuardianSetKeys(cell) {
|
|
230
240
|
const keys = [];
|
|
231
241
|
function parseCell(c) {
|
|
232
242
|
let slice = c.beginParse();
|
|
233
|
-
while
|
|
243
|
+
while(slice.remainingRefs > 0 || slice.remainingBits >= 160){
|
|
234
244
|
if (slice.remainingBits >= 160) {
|
|
235
245
|
const bitsToSkip = slice.remainingBits - 160;
|
|
236
246
|
slice = slice.skip(bitsToSkip);
|
|
@@ -245,8 +255,6 @@ function parseGuardianSetKeys(cell) {
|
|
|
245
255
|
parseCell(cell);
|
|
246
256
|
return keys;
|
|
247
257
|
}
|
|
248
|
-
function calculateUpdatePriceFeedsFee(numUpdates) {
|
|
249
|
-
return (
|
|
250
|
-
exports.UPDATE_PRICE_FEEDS_PER_UPDATE_GAS * numUpdates) *
|
|
251
|
-
exports.GAS_PRICE_FACTOR);
|
|
258
|
+
export function calculateUpdatePriceFeedsFee(numUpdates) {
|
|
259
|
+
return (UPDATE_PRICE_FEEDS_BASE_GAS + UPDATE_PRICE_FEEDS_PER_UPDATE_GAS * numUpdates) * GAS_PRICE_FACTOR;
|
|
252
260
|
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{ "type": "module" }
|
package/package.json
CHANGED
|
@@ -1,15 +1,15 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pythnetwork/pyth-ton-js",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.4.0",
|
|
4
4
|
"description": "Pyth Network TON Utilities",
|
|
5
5
|
"homepage": "https://pyth.network",
|
|
6
6
|
"author": {
|
|
7
7
|
"name": "Pyth Data Association"
|
|
8
8
|
},
|
|
9
|
-
"main": "
|
|
10
|
-
"types": "
|
|
9
|
+
"main": "./dist/cjs/index.cjs",
|
|
10
|
+
"types": "./dist/cjs/index.d.ts",
|
|
11
11
|
"files": [
|
|
12
|
-
"
|
|
12
|
+
"dist/**/*"
|
|
13
13
|
],
|
|
14
14
|
"repository": {
|
|
15
15
|
"type": "git",
|
|
@@ -25,26 +25,43 @@
|
|
|
25
25
|
],
|
|
26
26
|
"license": "Apache-2.0",
|
|
27
27
|
"devDependencies": {
|
|
28
|
+
"@cprussin/eslint-config": "^4.0.2",
|
|
28
29
|
"@ton/core": "^0.59.0",
|
|
29
30
|
"@ton/crypto": "^3.3.0",
|
|
30
31
|
"@ton/ton": "^15.1.0",
|
|
31
32
|
"@types/node": "^18.11.18",
|
|
32
|
-
"
|
|
33
|
-
"@typescript-eslint/parser": "^5.21.0",
|
|
34
|
-
"eslint": "^8.14.0",
|
|
33
|
+
"eslint": "^9.23.0",
|
|
35
34
|
"jest": "^29.4.1",
|
|
36
35
|
"prettier": "^3.5.3",
|
|
37
36
|
"ts-jest": "^29.0.5",
|
|
38
|
-
"ts-node": "^10.9.2"
|
|
39
|
-
"typescript": "^5.8.2"
|
|
37
|
+
"ts-node": "^10.9.2"
|
|
40
38
|
},
|
|
39
|
+
"engines": {
|
|
40
|
+
"node": "^24.0.0"
|
|
41
|
+
},
|
|
42
|
+
"type": "module",
|
|
43
|
+
"exports": {
|
|
44
|
+
".": {
|
|
45
|
+
"require": {
|
|
46
|
+
"types": "./dist/cjs/index.d.ts",
|
|
47
|
+
"default": "./dist/cjs/index.cjs"
|
|
48
|
+
},
|
|
49
|
+
"import": {
|
|
50
|
+
"types": "./dist/esm/index.d.ts",
|
|
51
|
+
"default": "./dist/esm/index.mjs"
|
|
52
|
+
}
|
|
53
|
+
},
|
|
54
|
+
"./package.json": "./package.json"
|
|
55
|
+
},
|
|
56
|
+
"module": "./dist/esm/index.mjs",
|
|
41
57
|
"scripts": {
|
|
42
|
-
"build": "
|
|
58
|
+
"build": "ts-duality",
|
|
43
59
|
"test:lint": "eslint src/ --max-warnings 0",
|
|
44
60
|
"test:format": "prettier --check \"src/**/*.ts\"",
|
|
45
61
|
"fix:lint": "eslint src/ --fix --max-warnings 0",
|
|
46
62
|
"fix:format": "prettier --write \"src/**/*.ts\"",
|
|
47
63
|
"preversion": "pnpm run test:lint",
|
|
48
|
-
"version": "pnpm run test:format && git add -A src"
|
|
64
|
+
"version": "pnpm run test:format && git add -A src",
|
|
65
|
+
"clean": "rm -rf ./dist"
|
|
49
66
|
}
|
|
50
67
|
}
|
package/lib/index.d.ts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,OAAO,EAEP,IAAI,EACJ,QAAQ,EAER,MAAM,EAGP,MAAM,WAAW,CAAC;AACnB,OAAO,EAAE,gBAAgB,EAAE,MAAM,UAAU,CAAC;AAE5C,eAAO,MAAM,6BAA6B,qDACU,CAAC;AACrD,eAAO,MAAM,6BAA6B,qDACU,CAAC;AAErD,eAAO,MAAM,2BAA2B,UAAU,CAAC;AACnD,eAAO,MAAM,iCAAiC,SAAS,CAAC;AAExD,eAAO,MAAM,gBAAgB,OAAO,CAAC;AAErC,MAAM,WAAW,UAAU;IACzB,YAAY,EAAE,MAAM,CAAC;IACrB,cAAc,EAAE,MAAM,CAAC;CACxB;AAED,qBAAa,YAAa,YAAW,QAAQ;IAEzC,QAAQ,CAAC,OAAO,EAAE,OAAO;IACzB,QAAQ,CAAC,IAAI,CAAC,EAAE;QAAE,IAAI,EAAE,IAAI,CAAC;QAAC,IAAI,EAAE,IAAI,CAAA;KAAE;gBADjC,OAAO,EAAE,OAAO,EAChB,IAAI,CAAC,EAAE;QAAE,IAAI,EAAE,IAAI,CAAC;QAAC,IAAI,EAAE,IAAI,CAAA;KAAE,YAAA;IAG5C,MAAM,CAAC,iBAAiB,CAAC,OAAO,EAAE,OAAO;IAInC,0BAA0B,CAAC,QAAQ,EAAE,gBAAgB;IAMrD,qBAAqB,CACzB,QAAQ,EAAE,gBAAgB,EAC1B,GAAG,EAAE,MAAM,EACX,EAAE,EAAE,MAAM;IAcN,oBAAoB,CACxB,QAAQ,EAAE,gBAAgB,EAC1B,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,MAAM;IAcb,2BAA2B,CAC/B,QAAQ,EAAE,gBAAgB,EAC1B,GAAG,EAAE,MAAM,EACX,gBAAgB,EAAE,MAAM;IAcpB,mBAAmB,CACvB,QAAQ,EAAE,gBAAgB,EAC1B,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,IAAI;IAcT,cAAc,CAAC,QAAQ,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM;;;;;;IAkB9D,mBAAmB,CACvB,QAAQ,EAAE,gBAAgB,EAC1B,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,MAAM;;;;;;IAoBf,iBAAiB,CAAC,QAAQ,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM;;;;;;IAkBjE,sBAAsB,CAC1B,QAAQ,EAAE,gBAAgB,EAC1B,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,MAAM;;;;;;IAoBf,YAAY,CAAC,QAAQ,EAAE,gBAAgB,EAAE,EAAE,EAAE,MAAM;IAQnD,kBAAkB,CAAC,QAAQ,EAAE,gBAAgB;IAM7C,iCAAiC,CAAC,QAAQ,EAAE,gBAAgB;IAS5D,UAAU,CAAC,QAAQ,EAAE,gBAAgB;IAMrC,cAAc,CAAC,QAAQ,EAAE,gBAAgB;IAKzC,uBAAuB,CAAC,QAAQ,EAAE,gBAAgB;IAKlD,cAAc,CAAC,QAAQ,EAAE,gBAAgB,EAAE,KAAK,EAAE,MAAM;;;;;CAe/D;AAED,wBAAgB,eAAe,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI,CAsBpD;AAmBD,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,IAAI,GAAG,UAAU,EAAE,CAczD;AAED,wBAAgB,eAAe,CAAC,IAAI,EAAE,IAAI,GAAG,UAAU,GAAG,IAAI,CAQ7D;AAED,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,IAAI,GAAG,MAAM,EAAE,CAoBzD;AAED,wBAAgB,4BAA4B,CAAC,UAAU,EAAE,MAAM,UAM9D"}
|