@clarigen/test 1.0.0-next.18 → 1.0.0-next.22
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/index.d.ts +2 -1
- package/dist/index.js +392 -2
- package/dist/index.mjs +392 -2
- package/package.json +19 -17
package/dist/index.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { CoreNodeEvent, ResultAssets, Contracts, ContractCall, Response, ContractCalls, ContractInstances } from '@clarigen/core';
|
|
1
|
+
import { CoreNodeEvent, ResultAssets, Contracts, ContractFactory, AllContracts, ContractCall, Response, ContractCalls, ContractInstances } from '@clarigen/core';
|
|
2
2
|
import { Costs, NativeClarityBinProvider, ClarinetAccounts } from '@clarigen/native-bin';
|
|
3
3
|
export { Allocation, createClarityBin, evalJson, executeJson } from '@clarigen/native-bin';
|
|
4
4
|
import { ClarityValue, ResponseCV } from 'micro-stacks/clarity';
|
|
@@ -57,6 +57,7 @@ declare class TestProvider {
|
|
|
57
57
|
coverageFolder?: string;
|
|
58
58
|
constructor(clarityBin: NativeClarityBinProvider);
|
|
59
59
|
static fromContracts<T extends Contracts<any>>(contracts: T, options?: FromContractsOptions): Promise<FromContracts<T>>;
|
|
60
|
+
static fromFactory(contracts: ContractFactory<AllContracts>, options?: FromContractsOptions): Promise<TestProvider>;
|
|
60
61
|
ro<T>(tx: ContractCall<T>): Promise<ReadOnlyResult<T>>;
|
|
61
62
|
rov<T>(tx: ContractCall<T>): Promise<T>;
|
|
62
63
|
roOk<Ok, Err>(tx: ContractCall<Response<Ok, Err>>): Promise<ReadOnlyResult<Ok>>;
|
package/dist/index.js
CHANGED
|
@@ -1,2 +1,392 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports, "__esModule", {value: true});var
|
|
2
|
-
|
|
1
|
+
"use strict";Object.defineProperty(exports, "__esModule", {value: true});var __defProp = Object.defineProperty;
|
|
2
|
+
var __defProps = Object.defineProperties;
|
|
3
|
+
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
|
4
|
+
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
|
7
|
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
8
|
+
var __spreadValues = (a, b) => {
|
|
9
|
+
for (var prop in b || (b = {}))
|
|
10
|
+
if (__hasOwnProp.call(b, prop))
|
|
11
|
+
__defNormalProp(a, prop, b[prop]);
|
|
12
|
+
if (__getOwnPropSymbols)
|
|
13
|
+
for (var prop of __getOwnPropSymbols(b)) {
|
|
14
|
+
if (__propIsEnum.call(b, prop))
|
|
15
|
+
__defNormalProp(a, prop, b[prop]);
|
|
16
|
+
}
|
|
17
|
+
return a;
|
|
18
|
+
};
|
|
19
|
+
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
|
|
20
|
+
|
|
21
|
+
// src/index.ts
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
var _nativebin = require('@clarigen/native-bin');
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
var _core = require('@clarigen/core');
|
|
31
|
+
|
|
32
|
+
// src/utils/index.ts
|
|
33
|
+
var _crypto = require('micro-stacks/crypto');
|
|
34
|
+
var _transactions = require('micro-stacks/transactions');
|
|
35
|
+
|
|
36
|
+
// src/utils/util-contract.ts
|
|
37
|
+
var _clarity = require('micro-stacks/clarity');
|
|
38
|
+
var _path = require('path');
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
var _common = require('micro-stacks/common');
|
|
42
|
+
var UTIL_CONTRACT_ID = "ST000000000000000000002AMW42H.clarigen-test-utils";
|
|
43
|
+
async function deployUtilContract(clarityBin) {
|
|
44
|
+
let contractFilePath = _path.join.call(void 0, __dirname, "..", "..", "contracts", "test-utils.clar");
|
|
45
|
+
if (__dirname.includes("dist")) {
|
|
46
|
+
contractFilePath = _path.join.call(void 0, __dirname, "..", "contracts", "test-utils.clar");
|
|
47
|
+
}
|
|
48
|
+
await _nativebin.deployContract.call(void 0, {
|
|
49
|
+
contractIdentifier: UTIL_CONTRACT_ID,
|
|
50
|
+
provider: clarityBin,
|
|
51
|
+
contractFilePath
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
function getBin(provider) {
|
|
55
|
+
return "clarityBin" in provider ? provider.clarityBin : provider;
|
|
56
|
+
}
|
|
57
|
+
async function getBlockHeight(provider) {
|
|
58
|
+
const bin = getBin(provider);
|
|
59
|
+
const { output_serialized } = await _nativebin.evalJson.call(void 0, {
|
|
60
|
+
contractAddress: UTIL_CONTRACT_ID,
|
|
61
|
+
functionName: "get-block-height",
|
|
62
|
+
args: [],
|
|
63
|
+
provider: bin
|
|
64
|
+
});
|
|
65
|
+
const outputCV = _clarity.hexToCV.call(void 0, output_serialized);
|
|
66
|
+
const blockHeight = _core.cvToValue.call(void 0, outputCV);
|
|
67
|
+
return blockHeight;
|
|
68
|
+
}
|
|
69
|
+
async function mineBlock(provider) {
|
|
70
|
+
const bin = getBin(provider);
|
|
71
|
+
await _nativebin.executeJson.call(void 0, {
|
|
72
|
+
contractAddress: UTIL_CONTRACT_ID,
|
|
73
|
+
functionName: "mine-block",
|
|
74
|
+
args: [],
|
|
75
|
+
provider: bin,
|
|
76
|
+
senderAddress: "ST000000000000000000002AMW42H"
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
async function mineBlocks(_blocks, provider) {
|
|
80
|
+
const blocks = _common.intToBigInt.call(void 0, _blocks);
|
|
81
|
+
for (let index = 0; index < blocks; index++) {
|
|
82
|
+
await mineBlock(provider);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
async function getStxBalance(provider, account) {
|
|
86
|
+
const bin = getBin(provider);
|
|
87
|
+
const { output_serialized } = await _nativebin.evalJson.call(void 0, {
|
|
88
|
+
contractAddress: UTIL_CONTRACT_ID,
|
|
89
|
+
functionName: "get-stx-balance",
|
|
90
|
+
args: [`'${account}`],
|
|
91
|
+
provider: bin
|
|
92
|
+
});
|
|
93
|
+
const outputCV = _clarity.hexToCV.call(void 0, output_serialized);
|
|
94
|
+
const balance = _core.cvToValue.call(void 0, outputCV);
|
|
95
|
+
return balance;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// src/utils/coverage.ts
|
|
99
|
+
var _promises = require('fs/promises');
|
|
100
|
+
|
|
101
|
+
async function setupCoverage(dir) {
|
|
102
|
+
const dirName = _path.resolve.call(void 0, process.cwd(), "coverage");
|
|
103
|
+
try {
|
|
104
|
+
await _promises.mkdir.call(void 0, dirName);
|
|
105
|
+
} catch (error) {
|
|
106
|
+
}
|
|
107
|
+
const files = await _promises.readdir.call(void 0, dirName);
|
|
108
|
+
const removals = files.map((file) => {
|
|
109
|
+
return _promises.unlink.call(void 0, _path.join.call(void 0, dirName, file));
|
|
110
|
+
});
|
|
111
|
+
await Promise.all(removals);
|
|
112
|
+
return dirName;
|
|
113
|
+
}
|
|
114
|
+
async function finishCoverage(provider, dir) {
|
|
115
|
+
const lcovFile = _path.join.call(void 0, dir, "clarigen.lcov");
|
|
116
|
+
await provider.runCommand(["make_lcov", dir, lcovFile]);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// src/utils/index.ts
|
|
120
|
+
function makeRandomAddress(version = _crypto.StacksNetworkVersion.testnetP2PKH) {
|
|
121
|
+
const privateKey = _transactions.makeRandomPrivKey.call(void 0, );
|
|
122
|
+
const publicKey = _transactions.getPublicKeyFromStacksPrivateKey.call(void 0, privateKey);
|
|
123
|
+
const address = _crypto.publicKeyToStxAddress.call(void 0, publicKey.data, version);
|
|
124
|
+
return address;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// src/utils/pure.ts
|
|
128
|
+
|
|
129
|
+
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
|
|
133
|
+
|
|
134
|
+
|
|
135
|
+
|
|
136
|
+
|
|
137
|
+
|
|
138
|
+
|
|
139
|
+
|
|
140
|
+
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
|
|
144
|
+
function getPrints(events) {
|
|
145
|
+
return _core.filterEvents.call(void 0, events, _core.CoreNodeEventType.ContractEvent).map((e) => {
|
|
146
|
+
const hex = e.contract_event.raw_value;
|
|
147
|
+
const cv = _clarity.hexToCV.call(void 0, hex);
|
|
148
|
+
return _core.cvToValue.call(void 0, cv);
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
function formatArguments(args) {
|
|
152
|
+
return args.map((arg) => _core.cvToString.call(void 0, arg));
|
|
153
|
+
}
|
|
154
|
+
function getIdentifier(tx2) {
|
|
155
|
+
return `${tx2.contractAddress}.${tx2.contractName}`;
|
|
156
|
+
}
|
|
157
|
+
function getLogs(stderr) {
|
|
158
|
+
if (stderr === "")
|
|
159
|
+
return [];
|
|
160
|
+
return stderr.split("\n").map((line) => line.slice(62));
|
|
161
|
+
}
|
|
162
|
+
function makeEvalResult(result) {
|
|
163
|
+
const resultCV = _clarity.hexToCV.call(void 0, result.output_serialized);
|
|
164
|
+
const value = _core.cvToValue.call(void 0, resultCV, true);
|
|
165
|
+
return {
|
|
166
|
+
value,
|
|
167
|
+
clarityValue: resultCV,
|
|
168
|
+
logs: getLogs(result.stderr),
|
|
169
|
+
costs: result.costs
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
async function ro({
|
|
173
|
+
tx: tx2,
|
|
174
|
+
bin,
|
|
175
|
+
coverageFolder
|
|
176
|
+
}) {
|
|
177
|
+
const result = await _nativebin.evalJson.call(void 0, {
|
|
178
|
+
functionName: tx2.function.name,
|
|
179
|
+
args: formatArguments(tx2.functionArgs),
|
|
180
|
+
contractAddress: getIdentifier(tx2),
|
|
181
|
+
provider: bin,
|
|
182
|
+
coverageFolder
|
|
183
|
+
});
|
|
184
|
+
return makeEvalResult(result);
|
|
185
|
+
}
|
|
186
|
+
async function evalCode({
|
|
187
|
+
contractAddress,
|
|
188
|
+
code,
|
|
189
|
+
bin,
|
|
190
|
+
coverageFolder
|
|
191
|
+
}) {
|
|
192
|
+
const result = await _nativebin.evalRaw.call(void 0, {
|
|
193
|
+
contractAddress,
|
|
194
|
+
code,
|
|
195
|
+
provider: bin,
|
|
196
|
+
coverageFolder
|
|
197
|
+
});
|
|
198
|
+
return makeEvalResult(result);
|
|
199
|
+
}
|
|
200
|
+
async function tx({
|
|
201
|
+
tx: tx2,
|
|
202
|
+
senderAddress,
|
|
203
|
+
bin,
|
|
204
|
+
coverageFolder
|
|
205
|
+
}) {
|
|
206
|
+
const result = await _nativebin.executeJson.call(void 0, {
|
|
207
|
+
contractAddress: getIdentifier(tx2),
|
|
208
|
+
senderAddress,
|
|
209
|
+
provider: bin,
|
|
210
|
+
functionName: tx2.function.name,
|
|
211
|
+
args: formatArguments(tx2.functionArgs),
|
|
212
|
+
coverageFolder
|
|
213
|
+
});
|
|
214
|
+
const resultCV = _clarity.hexToCV.call(void 0, result.output_serialized);
|
|
215
|
+
const value = _core.cvToValue.call(void 0, resultCV);
|
|
216
|
+
const baseReturn = {
|
|
217
|
+
value,
|
|
218
|
+
logs: getLogs(result.stderr),
|
|
219
|
+
costs: result.costs
|
|
220
|
+
};
|
|
221
|
+
if (result.success) {
|
|
222
|
+
return __spreadProps(__spreadValues({}, baseReturn), {
|
|
223
|
+
isOk: true,
|
|
224
|
+
response: _clarity.responseOkCV.call(void 0, resultCV),
|
|
225
|
+
assets: result.assets,
|
|
226
|
+
events: result.events,
|
|
227
|
+
prints: getPrints(result.events)
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
return __spreadProps(__spreadValues({}, baseReturn), {
|
|
231
|
+
isOk: false,
|
|
232
|
+
response: _clarity.responseErrorCV.call(void 0, resultCV)
|
|
233
|
+
});
|
|
234
|
+
}
|
|
235
|
+
async function mapGet({
|
|
236
|
+
map,
|
|
237
|
+
bin
|
|
238
|
+
}) {
|
|
239
|
+
const code = `(map-get? ${map.map.name} ${_core.cvToString.call(void 0, map.key)})`;
|
|
240
|
+
const result = await evalCode({
|
|
241
|
+
contractAddress: getIdentifier(map),
|
|
242
|
+
code,
|
|
243
|
+
bin
|
|
244
|
+
});
|
|
245
|
+
return result.value;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
// src/index.ts
|
|
249
|
+
|
|
250
|
+
|
|
251
|
+
var TestProvider = class {
|
|
252
|
+
constructor(clarityBin) {
|
|
253
|
+
this.clarityBin = clarityBin;
|
|
254
|
+
}
|
|
255
|
+
static async fromContracts(contracts, options = {}) {
|
|
256
|
+
const clarityBin = options.clarityBin || await _nativebin.createClarityBin.call(void 0, {
|
|
257
|
+
allocations: options.accounts
|
|
258
|
+
});
|
|
259
|
+
const instances = {};
|
|
260
|
+
let coverageFolder = options.coverageFolder;
|
|
261
|
+
if (process.env.CLARIGEN_COVERAGE) {
|
|
262
|
+
coverageFolder = _path.resolve.call(void 0, process.cwd(), "coverage");
|
|
263
|
+
}
|
|
264
|
+
await deployUtilContract(clarityBin);
|
|
265
|
+
for (const k in contracts) {
|
|
266
|
+
if (Object.prototype.hasOwnProperty.call(contracts, k)) {
|
|
267
|
+
const contract = contracts[k];
|
|
268
|
+
const identifier = _core.getContractIdentifier.call(void 0, contract);
|
|
269
|
+
await _nativebin.deployContract.call(void 0, {
|
|
270
|
+
contractIdentifier: identifier,
|
|
271
|
+
contractFilePath: contract.contractFile,
|
|
272
|
+
provider: clarityBin,
|
|
273
|
+
coverageFolder
|
|
274
|
+
});
|
|
275
|
+
const instance = contract.contract(contract.address, contract.name);
|
|
276
|
+
instances[k] = {
|
|
277
|
+
identifier: _core.getContractIdentifier.call(void 0, contract),
|
|
278
|
+
contract: instance
|
|
279
|
+
};
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
const provider = new this(clarityBin);
|
|
283
|
+
provider.coverageFolder = coverageFolder;
|
|
284
|
+
return {
|
|
285
|
+
deployed: instances,
|
|
286
|
+
provider
|
|
287
|
+
};
|
|
288
|
+
}
|
|
289
|
+
static async fromFactory(contracts, options = {}) {
|
|
290
|
+
const clarityBin = options.clarityBin || await _nativebin.createClarityBin.call(void 0, {
|
|
291
|
+
allocations: options.accounts
|
|
292
|
+
});
|
|
293
|
+
let coverageFolder = options.coverageFolder;
|
|
294
|
+
if (process.env.CLARIGEN_COVERAGE) {
|
|
295
|
+
coverageFolder = _path.resolve.call(void 0, process.cwd(), "coverage");
|
|
296
|
+
}
|
|
297
|
+
await deployUtilContract(clarityBin);
|
|
298
|
+
for (const k in contracts) {
|
|
299
|
+
if (Object.prototype.hasOwnProperty.call(contracts, k)) {
|
|
300
|
+
const contract = contracts[k];
|
|
301
|
+
const contractFilePath = contract.contractFile;
|
|
302
|
+
if (typeof contractFilePath === "undefined") {
|
|
303
|
+
throw new Error("Cannot setup @clarigen/test - missing contract file.");
|
|
304
|
+
}
|
|
305
|
+
await _nativebin.deployContract.call(void 0, {
|
|
306
|
+
contractIdentifier: contract.identifier,
|
|
307
|
+
contractFilePath,
|
|
308
|
+
provider: clarityBin,
|
|
309
|
+
coverageFolder
|
|
310
|
+
});
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
const provider = new this(clarityBin);
|
|
314
|
+
provider.coverageFolder = coverageFolder;
|
|
315
|
+
return provider;
|
|
316
|
+
}
|
|
317
|
+
ro(tx2) {
|
|
318
|
+
return ro({ tx: tx2, bin: this.clarityBin, coverageFolder: this.coverageFolder });
|
|
319
|
+
}
|
|
320
|
+
async rov(tx2) {
|
|
321
|
+
const result = await this.ro(tx2);
|
|
322
|
+
return result.value;
|
|
323
|
+
}
|
|
324
|
+
async roOk(tx2) {
|
|
325
|
+
const result = await this.ro(tx2);
|
|
326
|
+
const value = _core.expectOk.call(void 0, result.value);
|
|
327
|
+
return __spreadProps(__spreadValues({}, result), {
|
|
328
|
+
value
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
async roErr(tx2) {
|
|
332
|
+
const result = await this.ro(tx2);
|
|
333
|
+
const value = _core.expectErr.call(void 0, result.value);
|
|
334
|
+
return __spreadProps(__spreadValues({}, result), {
|
|
335
|
+
value
|
|
336
|
+
});
|
|
337
|
+
}
|
|
338
|
+
async rovOk(tx2) {
|
|
339
|
+
return (await this.roOk(tx2)).value;
|
|
340
|
+
}
|
|
341
|
+
async rovErr(tx2) {
|
|
342
|
+
return (await this.roErr(tx2)).value;
|
|
343
|
+
}
|
|
344
|
+
tx(tx2, senderAddress) {
|
|
345
|
+
return tx({
|
|
346
|
+
tx: tx2,
|
|
347
|
+
senderAddress,
|
|
348
|
+
bin: this.clarityBin,
|
|
349
|
+
coverageFolder: this.coverageFolder
|
|
350
|
+
});
|
|
351
|
+
}
|
|
352
|
+
async txOk(tx2, senderAddress) {
|
|
353
|
+
const result = await this.tx(tx2, senderAddress);
|
|
354
|
+
if (!result.isOk)
|
|
355
|
+
throw new Error(`Expected OK, received error: ${String(result.value)}`);
|
|
356
|
+
return result;
|
|
357
|
+
}
|
|
358
|
+
async txErr(tx2, senderAddress) {
|
|
359
|
+
const result = await this.tx(tx2, senderAddress);
|
|
360
|
+
if (result.isOk)
|
|
361
|
+
throw new Error(`Expected Err, received ok: ${String(result.value)}`);
|
|
362
|
+
return result;
|
|
363
|
+
}
|
|
364
|
+
evalCode(code, contractAddress = UTIL_CONTRACT_ID) {
|
|
365
|
+
return evalCode({
|
|
366
|
+
contractAddress,
|
|
367
|
+
code,
|
|
368
|
+
bin: this.clarityBin,
|
|
369
|
+
coverageFolder: this.coverageFolder
|
|
370
|
+
});
|
|
371
|
+
}
|
|
372
|
+
async eval(code, contractAddress = UTIL_CONTRACT_ID) {
|
|
373
|
+
const result = await this.evalCode(code, contractAddress);
|
|
374
|
+
return result.value;
|
|
375
|
+
}
|
|
376
|
+
mapGet(map) {
|
|
377
|
+
return mapGet({ map, bin: this.clarityBin });
|
|
378
|
+
}
|
|
379
|
+
};
|
|
380
|
+
|
|
381
|
+
|
|
382
|
+
|
|
383
|
+
|
|
384
|
+
|
|
385
|
+
|
|
386
|
+
|
|
387
|
+
|
|
388
|
+
|
|
389
|
+
|
|
390
|
+
|
|
391
|
+
|
|
392
|
+
exports.Allocation = _nativebin.Allocation; exports.TestProvider = TestProvider; exports.createClarityBin = _nativebin.createClarityBin; exports.evalJson = _nativebin.evalJson; exports.executeJson = _nativebin.executeJson; exports.finishCoverage = finishCoverage; exports.getBlockHeight = getBlockHeight; exports.getStxBalance = getStxBalance; exports.makeRandomAddress = makeRandomAddress; exports.mineBlocks = mineBlocks; exports.setupCoverage = setupCoverage;
|
package/dist/index.mjs
CHANGED
|
@@ -1,2 +1,392 @@
|
|
|
1
|
-
var
|
|
2
|
-
|
|
1
|
+
var __defProp = Object.defineProperty;
|
|
2
|
+
var __defProps = Object.defineProperties;
|
|
3
|
+
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
|
4
|
+
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
|
7
|
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
8
|
+
var __spreadValues = (a, b) => {
|
|
9
|
+
for (var prop in b || (b = {}))
|
|
10
|
+
if (__hasOwnProp.call(b, prop))
|
|
11
|
+
__defNormalProp(a, prop, b[prop]);
|
|
12
|
+
if (__getOwnPropSymbols)
|
|
13
|
+
for (var prop of __getOwnPropSymbols(b)) {
|
|
14
|
+
if (__propIsEnum.call(b, prop))
|
|
15
|
+
__defNormalProp(a, prop, b[prop]);
|
|
16
|
+
}
|
|
17
|
+
return a;
|
|
18
|
+
};
|
|
19
|
+
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
|
|
20
|
+
|
|
21
|
+
// src/index.ts
|
|
22
|
+
import {
|
|
23
|
+
deployContract as deployContract2,
|
|
24
|
+
createClarityBin
|
|
25
|
+
} from "@clarigen/native-bin";
|
|
26
|
+
import {
|
|
27
|
+
getContractIdentifier,
|
|
28
|
+
expectOk,
|
|
29
|
+
expectErr
|
|
30
|
+
} from "@clarigen/core";
|
|
31
|
+
|
|
32
|
+
// src/utils/index.ts
|
|
33
|
+
import { publicKeyToStxAddress, StacksNetworkVersion } from "micro-stacks/crypto";
|
|
34
|
+
import { getPublicKeyFromStacksPrivateKey, makeRandomPrivKey } from "micro-stacks/transactions";
|
|
35
|
+
|
|
36
|
+
// src/utils/util-contract.ts
|
|
37
|
+
import { hexToCV } from "micro-stacks/clarity";
|
|
38
|
+
import { join } from "path";
|
|
39
|
+
import { cvToValue } from "@clarigen/core";
|
|
40
|
+
import { deployContract, evalJson, executeJson } from "@clarigen/native-bin";
|
|
41
|
+
import { intToBigInt } from "micro-stacks/common";
|
|
42
|
+
var UTIL_CONTRACT_ID = "ST000000000000000000002AMW42H.clarigen-test-utils";
|
|
43
|
+
async function deployUtilContract(clarityBin) {
|
|
44
|
+
let contractFilePath = join(__dirname, "..", "..", "contracts", "test-utils.clar");
|
|
45
|
+
if (__dirname.includes("dist")) {
|
|
46
|
+
contractFilePath = join(__dirname, "..", "contracts", "test-utils.clar");
|
|
47
|
+
}
|
|
48
|
+
await deployContract({
|
|
49
|
+
contractIdentifier: UTIL_CONTRACT_ID,
|
|
50
|
+
provider: clarityBin,
|
|
51
|
+
contractFilePath
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
function getBin(provider) {
|
|
55
|
+
return "clarityBin" in provider ? provider.clarityBin : provider;
|
|
56
|
+
}
|
|
57
|
+
async function getBlockHeight(provider) {
|
|
58
|
+
const bin = getBin(provider);
|
|
59
|
+
const { output_serialized } = await evalJson({
|
|
60
|
+
contractAddress: UTIL_CONTRACT_ID,
|
|
61
|
+
functionName: "get-block-height",
|
|
62
|
+
args: [],
|
|
63
|
+
provider: bin
|
|
64
|
+
});
|
|
65
|
+
const outputCV = hexToCV(output_serialized);
|
|
66
|
+
const blockHeight = cvToValue(outputCV);
|
|
67
|
+
return blockHeight;
|
|
68
|
+
}
|
|
69
|
+
async function mineBlock(provider) {
|
|
70
|
+
const bin = getBin(provider);
|
|
71
|
+
await executeJson({
|
|
72
|
+
contractAddress: UTIL_CONTRACT_ID,
|
|
73
|
+
functionName: "mine-block",
|
|
74
|
+
args: [],
|
|
75
|
+
provider: bin,
|
|
76
|
+
senderAddress: "ST000000000000000000002AMW42H"
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
async function mineBlocks(_blocks, provider) {
|
|
80
|
+
const blocks = intToBigInt(_blocks);
|
|
81
|
+
for (let index = 0; index < blocks; index++) {
|
|
82
|
+
await mineBlock(provider);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
async function getStxBalance(provider, account) {
|
|
86
|
+
const bin = getBin(provider);
|
|
87
|
+
const { output_serialized } = await evalJson({
|
|
88
|
+
contractAddress: UTIL_CONTRACT_ID,
|
|
89
|
+
functionName: "get-stx-balance",
|
|
90
|
+
args: [`'${account}`],
|
|
91
|
+
provider: bin
|
|
92
|
+
});
|
|
93
|
+
const outputCV = hexToCV(output_serialized);
|
|
94
|
+
const balance = cvToValue(outputCV);
|
|
95
|
+
return balance;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// src/utils/coverage.ts
|
|
99
|
+
import { mkdir, readdir, unlink } from "fs/promises";
|
|
100
|
+
import { join as join2, resolve } from "path";
|
|
101
|
+
async function setupCoverage(dir) {
|
|
102
|
+
const dirName = resolve(process.cwd(), "coverage");
|
|
103
|
+
try {
|
|
104
|
+
await mkdir(dirName);
|
|
105
|
+
} catch (error) {
|
|
106
|
+
}
|
|
107
|
+
const files = await readdir(dirName);
|
|
108
|
+
const removals = files.map((file) => {
|
|
109
|
+
return unlink(join2(dirName, file));
|
|
110
|
+
});
|
|
111
|
+
await Promise.all(removals);
|
|
112
|
+
return dirName;
|
|
113
|
+
}
|
|
114
|
+
async function finishCoverage(provider, dir) {
|
|
115
|
+
const lcovFile = join2(dir, "clarigen.lcov");
|
|
116
|
+
await provider.runCommand(["make_lcov", dir, lcovFile]);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// src/utils/index.ts
|
|
120
|
+
function makeRandomAddress(version = StacksNetworkVersion.testnetP2PKH) {
|
|
121
|
+
const privateKey = makeRandomPrivKey();
|
|
122
|
+
const publicKey = getPublicKeyFromStacksPrivateKey(privateKey);
|
|
123
|
+
const address = publicKeyToStxAddress(publicKey.data, version);
|
|
124
|
+
return address;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// src/utils/pure.ts
|
|
128
|
+
import {
|
|
129
|
+
cvToString,
|
|
130
|
+
cvToValue as cvToValue2,
|
|
131
|
+
filterEvents,
|
|
132
|
+
CoreNodeEventType
|
|
133
|
+
} from "@clarigen/core";
|
|
134
|
+
import {
|
|
135
|
+
evalRaw,
|
|
136
|
+
evalJson as evalJson2,
|
|
137
|
+
executeJson as executeJson2
|
|
138
|
+
} from "@clarigen/native-bin";
|
|
139
|
+
import {
|
|
140
|
+
hexToCV as hexToCV2,
|
|
141
|
+
responseErrorCV,
|
|
142
|
+
responseOkCV
|
|
143
|
+
} from "micro-stacks/clarity";
|
|
144
|
+
function getPrints(events) {
|
|
145
|
+
return filterEvents(events, CoreNodeEventType.ContractEvent).map((e) => {
|
|
146
|
+
const hex = e.contract_event.raw_value;
|
|
147
|
+
const cv = hexToCV2(hex);
|
|
148
|
+
return cvToValue2(cv);
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
function formatArguments(args) {
|
|
152
|
+
return args.map((arg) => cvToString(arg));
|
|
153
|
+
}
|
|
154
|
+
function getIdentifier(tx2) {
|
|
155
|
+
return `${tx2.contractAddress}.${tx2.contractName}`;
|
|
156
|
+
}
|
|
157
|
+
function getLogs(stderr) {
|
|
158
|
+
if (stderr === "")
|
|
159
|
+
return [];
|
|
160
|
+
return stderr.split("\n").map((line) => line.slice(62));
|
|
161
|
+
}
|
|
162
|
+
function makeEvalResult(result) {
|
|
163
|
+
const resultCV = hexToCV2(result.output_serialized);
|
|
164
|
+
const value = cvToValue2(resultCV, true);
|
|
165
|
+
return {
|
|
166
|
+
value,
|
|
167
|
+
clarityValue: resultCV,
|
|
168
|
+
logs: getLogs(result.stderr),
|
|
169
|
+
costs: result.costs
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
async function ro({
|
|
173
|
+
tx: tx2,
|
|
174
|
+
bin,
|
|
175
|
+
coverageFolder
|
|
176
|
+
}) {
|
|
177
|
+
const result = await evalJson2({
|
|
178
|
+
functionName: tx2.function.name,
|
|
179
|
+
args: formatArguments(tx2.functionArgs),
|
|
180
|
+
contractAddress: getIdentifier(tx2),
|
|
181
|
+
provider: bin,
|
|
182
|
+
coverageFolder
|
|
183
|
+
});
|
|
184
|
+
return makeEvalResult(result);
|
|
185
|
+
}
|
|
186
|
+
async function evalCode({
|
|
187
|
+
contractAddress,
|
|
188
|
+
code,
|
|
189
|
+
bin,
|
|
190
|
+
coverageFolder
|
|
191
|
+
}) {
|
|
192
|
+
const result = await evalRaw({
|
|
193
|
+
contractAddress,
|
|
194
|
+
code,
|
|
195
|
+
provider: bin,
|
|
196
|
+
coverageFolder
|
|
197
|
+
});
|
|
198
|
+
return makeEvalResult(result);
|
|
199
|
+
}
|
|
200
|
+
async function tx({
|
|
201
|
+
tx: tx2,
|
|
202
|
+
senderAddress,
|
|
203
|
+
bin,
|
|
204
|
+
coverageFolder
|
|
205
|
+
}) {
|
|
206
|
+
const result = await executeJson2({
|
|
207
|
+
contractAddress: getIdentifier(tx2),
|
|
208
|
+
senderAddress,
|
|
209
|
+
provider: bin,
|
|
210
|
+
functionName: tx2.function.name,
|
|
211
|
+
args: formatArguments(tx2.functionArgs),
|
|
212
|
+
coverageFolder
|
|
213
|
+
});
|
|
214
|
+
const resultCV = hexToCV2(result.output_serialized);
|
|
215
|
+
const value = cvToValue2(resultCV);
|
|
216
|
+
const baseReturn = {
|
|
217
|
+
value,
|
|
218
|
+
logs: getLogs(result.stderr),
|
|
219
|
+
costs: result.costs
|
|
220
|
+
};
|
|
221
|
+
if (result.success) {
|
|
222
|
+
return __spreadProps(__spreadValues({}, baseReturn), {
|
|
223
|
+
isOk: true,
|
|
224
|
+
response: responseOkCV(resultCV),
|
|
225
|
+
assets: result.assets,
|
|
226
|
+
events: result.events,
|
|
227
|
+
prints: getPrints(result.events)
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
return __spreadProps(__spreadValues({}, baseReturn), {
|
|
231
|
+
isOk: false,
|
|
232
|
+
response: responseErrorCV(resultCV)
|
|
233
|
+
});
|
|
234
|
+
}
|
|
235
|
+
async function mapGet({
|
|
236
|
+
map,
|
|
237
|
+
bin
|
|
238
|
+
}) {
|
|
239
|
+
const code = `(map-get? ${map.map.name} ${cvToString(map.key)})`;
|
|
240
|
+
const result = await evalCode({
|
|
241
|
+
contractAddress: getIdentifier(map),
|
|
242
|
+
code,
|
|
243
|
+
bin
|
|
244
|
+
});
|
|
245
|
+
return result.value;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
// src/index.ts
|
|
249
|
+
import { resolve as resolve2 } from "path";
|
|
250
|
+
import { Allocation, createClarityBin as createClarityBin2, executeJson as executeJson3, evalJson as evalJson3 } from "@clarigen/native-bin";
|
|
251
|
+
var TestProvider = class {
|
|
252
|
+
constructor(clarityBin) {
|
|
253
|
+
this.clarityBin = clarityBin;
|
|
254
|
+
}
|
|
255
|
+
static async fromContracts(contracts, options = {}) {
|
|
256
|
+
const clarityBin = options.clarityBin || await createClarityBin({
|
|
257
|
+
allocations: options.accounts
|
|
258
|
+
});
|
|
259
|
+
const instances = {};
|
|
260
|
+
let coverageFolder = options.coverageFolder;
|
|
261
|
+
if (process.env.CLARIGEN_COVERAGE) {
|
|
262
|
+
coverageFolder = resolve2(process.cwd(), "coverage");
|
|
263
|
+
}
|
|
264
|
+
await deployUtilContract(clarityBin);
|
|
265
|
+
for (const k in contracts) {
|
|
266
|
+
if (Object.prototype.hasOwnProperty.call(contracts, k)) {
|
|
267
|
+
const contract = contracts[k];
|
|
268
|
+
const identifier = getContractIdentifier(contract);
|
|
269
|
+
await deployContract2({
|
|
270
|
+
contractIdentifier: identifier,
|
|
271
|
+
contractFilePath: contract.contractFile,
|
|
272
|
+
provider: clarityBin,
|
|
273
|
+
coverageFolder
|
|
274
|
+
});
|
|
275
|
+
const instance = contract.contract(contract.address, contract.name);
|
|
276
|
+
instances[k] = {
|
|
277
|
+
identifier: getContractIdentifier(contract),
|
|
278
|
+
contract: instance
|
|
279
|
+
};
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
const provider = new this(clarityBin);
|
|
283
|
+
provider.coverageFolder = coverageFolder;
|
|
284
|
+
return {
|
|
285
|
+
deployed: instances,
|
|
286
|
+
provider
|
|
287
|
+
};
|
|
288
|
+
}
|
|
289
|
+
static async fromFactory(contracts, options = {}) {
|
|
290
|
+
const clarityBin = options.clarityBin || await createClarityBin({
|
|
291
|
+
allocations: options.accounts
|
|
292
|
+
});
|
|
293
|
+
let coverageFolder = options.coverageFolder;
|
|
294
|
+
if (process.env.CLARIGEN_COVERAGE) {
|
|
295
|
+
coverageFolder = resolve2(process.cwd(), "coverage");
|
|
296
|
+
}
|
|
297
|
+
await deployUtilContract(clarityBin);
|
|
298
|
+
for (const k in contracts) {
|
|
299
|
+
if (Object.prototype.hasOwnProperty.call(contracts, k)) {
|
|
300
|
+
const contract = contracts[k];
|
|
301
|
+
const contractFilePath = contract.contractFile;
|
|
302
|
+
if (typeof contractFilePath === "undefined") {
|
|
303
|
+
throw new Error("Cannot setup @clarigen/test - missing contract file.");
|
|
304
|
+
}
|
|
305
|
+
await deployContract2({
|
|
306
|
+
contractIdentifier: contract.identifier,
|
|
307
|
+
contractFilePath,
|
|
308
|
+
provider: clarityBin,
|
|
309
|
+
coverageFolder
|
|
310
|
+
});
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
const provider = new this(clarityBin);
|
|
314
|
+
provider.coverageFolder = coverageFolder;
|
|
315
|
+
return provider;
|
|
316
|
+
}
|
|
317
|
+
ro(tx2) {
|
|
318
|
+
return ro({ tx: tx2, bin: this.clarityBin, coverageFolder: this.coverageFolder });
|
|
319
|
+
}
|
|
320
|
+
async rov(tx2) {
|
|
321
|
+
const result = await this.ro(tx2);
|
|
322
|
+
return result.value;
|
|
323
|
+
}
|
|
324
|
+
async roOk(tx2) {
|
|
325
|
+
const result = await this.ro(tx2);
|
|
326
|
+
const value = expectOk(result.value);
|
|
327
|
+
return __spreadProps(__spreadValues({}, result), {
|
|
328
|
+
value
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
async roErr(tx2) {
|
|
332
|
+
const result = await this.ro(tx2);
|
|
333
|
+
const value = expectErr(result.value);
|
|
334
|
+
return __spreadProps(__spreadValues({}, result), {
|
|
335
|
+
value
|
|
336
|
+
});
|
|
337
|
+
}
|
|
338
|
+
async rovOk(tx2) {
|
|
339
|
+
return (await this.roOk(tx2)).value;
|
|
340
|
+
}
|
|
341
|
+
async rovErr(tx2) {
|
|
342
|
+
return (await this.roErr(tx2)).value;
|
|
343
|
+
}
|
|
344
|
+
tx(tx2, senderAddress) {
|
|
345
|
+
return tx({
|
|
346
|
+
tx: tx2,
|
|
347
|
+
senderAddress,
|
|
348
|
+
bin: this.clarityBin,
|
|
349
|
+
coverageFolder: this.coverageFolder
|
|
350
|
+
});
|
|
351
|
+
}
|
|
352
|
+
async txOk(tx2, senderAddress) {
|
|
353
|
+
const result = await this.tx(tx2, senderAddress);
|
|
354
|
+
if (!result.isOk)
|
|
355
|
+
throw new Error(`Expected OK, received error: ${String(result.value)}`);
|
|
356
|
+
return result;
|
|
357
|
+
}
|
|
358
|
+
async txErr(tx2, senderAddress) {
|
|
359
|
+
const result = await this.tx(tx2, senderAddress);
|
|
360
|
+
if (result.isOk)
|
|
361
|
+
throw new Error(`Expected Err, received ok: ${String(result.value)}`);
|
|
362
|
+
return result;
|
|
363
|
+
}
|
|
364
|
+
evalCode(code, contractAddress = UTIL_CONTRACT_ID) {
|
|
365
|
+
return evalCode({
|
|
366
|
+
contractAddress,
|
|
367
|
+
code,
|
|
368
|
+
bin: this.clarityBin,
|
|
369
|
+
coverageFolder: this.coverageFolder
|
|
370
|
+
});
|
|
371
|
+
}
|
|
372
|
+
async eval(code, contractAddress = UTIL_CONTRACT_ID) {
|
|
373
|
+
const result = await this.evalCode(code, contractAddress);
|
|
374
|
+
return result.value;
|
|
375
|
+
}
|
|
376
|
+
mapGet(map) {
|
|
377
|
+
return mapGet({ map, bin: this.clarityBin });
|
|
378
|
+
}
|
|
379
|
+
};
|
|
380
|
+
export {
|
|
381
|
+
Allocation,
|
|
382
|
+
TestProvider,
|
|
383
|
+
createClarityBin2 as createClarityBin,
|
|
384
|
+
evalJson3 as evalJson,
|
|
385
|
+
executeJson3 as executeJson,
|
|
386
|
+
finishCoverage,
|
|
387
|
+
getBlockHeight,
|
|
388
|
+
getStxBalance,
|
|
389
|
+
makeRandomAddress,
|
|
390
|
+
mineBlocks,
|
|
391
|
+
setupCoverage
|
|
392
|
+
};
|
package/package.json
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
{
|
|
2
|
-
"version": "1.0.0-next.
|
|
2
|
+
"version": "1.0.0-next.22",
|
|
3
3
|
"license": "MIT",
|
|
4
4
|
"types": "./dist/index.d.ts",
|
|
5
5
|
"main": "./dist/index.js",
|
|
@@ -12,27 +12,29 @@
|
|
|
12
12
|
"engines": {
|
|
13
13
|
"node": ">=10"
|
|
14
14
|
},
|
|
15
|
-
"scripts": {
|
|
16
|
-
"start": "tsup --watch",
|
|
17
|
-
"dev": "tsup --watch",
|
|
18
|
-
"build": "shx rm -rf ./dist && tsup-node",
|
|
19
|
-
"test": "yarn build-test-types && jest",
|
|
20
|
-
"lint": "eslint \"src/**/*.{ts,tsx}\" && prettier --check src/**/*.ts",
|
|
21
|
-
"prepublishOnly": "yarn build",
|
|
22
|
-
"typecheck": "tsc --noEmit",
|
|
23
|
-
"build-test-types": "cd test && yarn clarigen && cd ..",
|
|
24
|
-
"publish:dev": "yalc publish --push"
|
|
25
|
-
},
|
|
26
15
|
"prettier": "@stacks/prettier-config",
|
|
27
16
|
"name": "@clarigen/test",
|
|
28
17
|
"author": "Hank Stoever",
|
|
18
|
+
"devDependencies": {
|
|
19
|
+
"jest-environment-node": "28.1.1",
|
|
20
|
+
"@clarigen/cli": "1.0.0-next.22"
|
|
21
|
+
},
|
|
29
22
|
"dependencies": {
|
|
30
|
-
"@clarigen/core": "1.0.0-next.
|
|
31
|
-
"@clarigen/native-bin": "1.0.0-next.
|
|
32
|
-
"micro-stacks": "^0.2
|
|
23
|
+
"@clarigen/core": "1.0.0-next.22",
|
|
24
|
+
"@clarigen/native-bin": "1.0.0-next.22",
|
|
25
|
+
"micro-stacks": "^0.5.2"
|
|
33
26
|
},
|
|
34
27
|
"publishConfig": {
|
|
35
28
|
"access": "public"
|
|
36
29
|
},
|
|
37
|
-
"
|
|
38
|
-
|
|
30
|
+
"scripts": {
|
|
31
|
+
"start": "tsup-node --watch",
|
|
32
|
+
"dev": "tsup-node --watch",
|
|
33
|
+
"build": "shx rm -rf ./dist && tsup-node",
|
|
34
|
+
"test": "pnpm build-test-types && jest",
|
|
35
|
+
"lint": "eslint \"src/**/*.{ts,tsx}\" && prettier --check src/**/*.ts",
|
|
36
|
+
"typecheck": "tsc --noEmit",
|
|
37
|
+
"build-test-types": "cd test/clarinet-project && pnpm clarigen && cd ../..",
|
|
38
|
+
"publish:dev": "yalc publish --push"
|
|
39
|
+
}
|
|
40
|
+
}
|