Compare commits
	
		
			7 Commits
		
	
	
		
			master
			...
			ian/upstre
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 825b514c8b | ||
|  | 0db343a164 | ||
|  | 9a1004301a | ||
|  | 1f5f6ae0f2 | ||
|  | b3b1695ced | ||
|  | 36298e55e6 | ||
|  | b450ef684d | 
							
								
								
									
										15
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @ -1,15 +1,21 @@ | ||||
| # Lotus changelog | ||||
| 
 | ||||
| # UNRELEASED | ||||
| - chore: Auto remove local chain data when importing chain file or snapshot ([filecoin-project/lotus#11277](https://github.com/filecoin-project/lotus/pull/11277)) | ||||
| - feat: metric: export Mpool message count ([filecoin-project/lotus#11361](https://github.com/filecoin-project/lotus/pull/11361)) | ||||
| - feat: sealing: load SectorsSummary from sealing SectorStats instead of calling API each time ([filecoin-project/lotus#11353](https://github.com/filecoin-project/lotus/pull/11353)) | ||||
| - fix: miner info: Show correct sector state counts ([filecoin-project/lotus#11456](https://github.com/filecoin-project/lotus/pull/11456)) | ||||
| - feat: add support for specifying block number when calling `eth_estimateGas` ([filecoin-project/lotus#11462](https://github.com/filecoin-project/lotus/pull/11462)). | ||||
| 
 | ||||
| ## Improvements | ||||
| - fix: exchange: allow up to 10k messages per block ([filecoin-project/lotus#11506](https://github.com/filecoin-project/lotus/pull/11506)) | ||||
| 
 | ||||
| # v1.25.1 / 2023-12-09 | ||||
| 
 | ||||
| This is a **highly recommended PATCH RELEASE.**  The patch release fixes the issue were node operators trying to catch up sync were unable to sync large message blocks/epochs due to an increased number of messages on the network. | ||||
| 
 | ||||
| This patch release allows for up to 10k messages per block. Additionally, it introduces a limit on the amount of data that can be read at once, ensuring the system can handle worst-case scenarios. | ||||
| 
 | ||||
| ## Improvements | ||||
| - fix: exchange: allow up to 10k messages per block ([filecoin-project/lotus#11506](https://github.com/filecoin-project/lotus/pull/11506)) | ||||
| 
 | ||||
| 
 | ||||
| # v 1.25.0 / 2023-11-22 | ||||
| 
 | ||||
| This is a highly recommended feature release of Lotus. This optional release supports the Filecoin network version 21 upgrade, codenamed Watermelon 🍉, in addition to the numerous improvements and enhancements for node operators, ETH RPC-providers and storage providers. | ||||
| @ -94,6 +100,7 @@ Lotus-workers can now be built to leverage the SupraSeal C2 sealing optimization | ||||
| - fix(client): single-root error message ([filecoin-project/lotus#11214](https://github.com/filecoin-project/lotus/pull/11214)) | ||||
| - fix: worker: Convert `DC_[SectorSize]_[ResourceRestriction]` if set ([filecoin-project/lotus#11224](https://github.com/filecoin-project/lotus/pull/11224)) | ||||
| - chore: backport #11338 onto release/v1.25.0 ([filecoin-project/lotus#11350](https://github.com/filecoin-project/lotus/pull/11350)) | ||||
| - fix: Exclude reverted events in `eth_getLogs` results [filecoin-project/lotus#11318](https://github.com/filecoin-project/lotus/pull/11318) | ||||
| 
 | ||||
| ## Dependencies | ||||
| - deps: update go-libp2p to v0.28.1 ([filecoin-project/lotus#10998](https://github.com/filecoin-project/lotus/pull/10998)) | ||||
|  | ||||
| @ -824,7 +824,7 @@ type FullNode interface { | ||||
| 	EthFeeHistory(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthFeeHistory, error)                                                                          //perm:read
 | ||||
| 
 | ||||
| 	EthMaxPriorityFeePerGas(ctx context.Context) (ethtypes.EthBigInt, error)                                             //perm:read
 | ||||
| 	EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error)                                 //perm:read
 | ||||
| 	EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error)                                 //perm:read
 | ||||
| 	EthCall(ctx context.Context, tx ethtypes.EthCall, blkParam ethtypes.EthBlockNumberOrHash) (ethtypes.EthBytes, error) //perm:read
 | ||||
| 
 | ||||
| 	EthSendRawTransaction(ctx context.Context, rawTx ethtypes.EthBytes) (ethtypes.EthHash, error) //perm:read
 | ||||
|  | ||||
| @ -114,7 +114,7 @@ type Gateway interface { | ||||
| 	EthGasPrice(ctx context.Context) (ethtypes.EthBigInt, error) | ||||
| 	EthFeeHistory(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthFeeHistory, error) | ||||
| 	EthMaxPriorityFeePerGas(ctx context.Context) (ethtypes.EthBigInt, error) | ||||
| 	EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error) | ||||
| 	EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error) | ||||
| 	EthCall(ctx context.Context, tx ethtypes.EthCall, blkParam ethtypes.EthBlockNumberOrHash) (ethtypes.EthBytes, error) | ||||
| 	EthSendRawTransaction(ctx context.Context, rawTx ethtypes.EthBytes) (ethtypes.EthHash, error) | ||||
| 	EthGetLogs(ctx context.Context, filter *ethtypes.EthFilterSpec) (*ethtypes.EthFilterResult, error) | ||||
|  | ||||
| @ -1042,7 +1042,7 @@ func (mr *MockFullNodeMockRecorder) EthChainId(arg0 interface{}) *gomock.Call { | ||||
| } | ||||
| 
 | ||||
| // EthEstimateGas mocks base method.
 | ||||
| func (m *MockFullNode) EthEstimateGas(arg0 context.Context, arg1 ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (m *MockFullNode) EthEstimateGas(arg0 context.Context, arg1 jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	m.ctrl.T.Helper() | ||||
| 	ret := m.ctrl.Call(m, "EthEstimateGas", arg0, arg1) | ||||
| 	ret0, _ := ret[0].(ethtypes.EthUint64) | ||||
|  | ||||
| @ -255,7 +255,7 @@ type FullNodeMethods struct { | ||||
| 
 | ||||
| 	EthChainId func(p0 context.Context) (ethtypes.EthUint64, error) `perm:"read"` | ||||
| 
 | ||||
| 	EthEstimateGas func(p0 context.Context, p1 ethtypes.EthCall) (ethtypes.EthUint64, error) `perm:"read"` | ||||
| 	EthEstimateGas func(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthUint64, error) `perm:"read"` | ||||
| 
 | ||||
| 	EthFeeHistory func(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthFeeHistory, error) `perm:"read"` | ||||
| 
 | ||||
| @ -679,7 +679,7 @@ type GatewayMethods struct { | ||||
| 
 | ||||
| 	EthChainId func(p0 context.Context) (ethtypes.EthUint64, error) `` | ||||
| 
 | ||||
| 	EthEstimateGas func(p0 context.Context, p1 ethtypes.EthCall) (ethtypes.EthUint64, error) `` | ||||
| 	EthEstimateGas func(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthUint64, error) `` | ||||
| 
 | ||||
| 	EthFeeHistory func(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthFeeHistory, error) `` | ||||
| 
 | ||||
| @ -2134,14 +2134,14 @@ func (s *FullNodeStub) EthChainId(p0 context.Context) (ethtypes.EthUint64, error | ||||
| 	return *new(ethtypes.EthUint64), ErrNotSupported | ||||
| } | ||||
| 
 | ||||
| func (s *FullNodeStruct) EthEstimateGas(p0 context.Context, p1 ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (s *FullNodeStruct) EthEstimateGas(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	if s.Internal.EthEstimateGas == nil { | ||||
| 		return *new(ethtypes.EthUint64), ErrNotSupported | ||||
| 	} | ||||
| 	return s.Internal.EthEstimateGas(p0, p1) | ||||
| } | ||||
| 
 | ||||
| func (s *FullNodeStub) EthEstimateGas(p0 context.Context, p1 ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (s *FullNodeStub) EthEstimateGas(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	return *new(ethtypes.EthUint64), ErrNotSupported | ||||
| } | ||||
| 
 | ||||
| @ -4400,14 +4400,14 @@ func (s *GatewayStub) EthChainId(p0 context.Context) (ethtypes.EthUint64, error) | ||||
| 	return *new(ethtypes.EthUint64), ErrNotSupported | ||||
| } | ||||
| 
 | ||||
| func (s *GatewayStruct) EthEstimateGas(p0 context.Context, p1 ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (s *GatewayStruct) EthEstimateGas(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	if s.Internal.EthEstimateGas == nil { | ||||
| 		return *new(ethtypes.EthUint64), ErrNotSupported | ||||
| 	} | ||||
| 	return s.Internal.EthEstimateGas(p0, p1) | ||||
| } | ||||
| 
 | ||||
| func (s *GatewayStub) EthEstimateGas(p0 context.Context, p1 ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (s *GatewayStub) EthEstimateGas(p0 context.Context, p1 jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	return *new(ethtypes.EthUint64), ErrNotSupported | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -388,7 +388,7 @@ func (m *EventFilterManager) Install(ctx context.Context, minHeight, maxHeight a | ||||
| 
 | ||||
| 	if m.EventIndex != nil && minHeight != -1 && minHeight < currentHeight { | ||||
| 		// Filter needs historic events
 | ||||
| 		if err := m.EventIndex.PrefillFilter(ctx, f); err != nil { | ||||
| 		if err := m.EventIndex.PrefillFilter(ctx, f, true); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @ -46,6 +46,7 @@ var ddls = []string{ | ||||
| 	)`, | ||||
| 
 | ||||
| 	`CREATE INDEX IF NOT EXISTS height_tipset_key ON event (height,tipset_key)`, | ||||
| 	`CREATE INDEX IF NOT EXISTS event_emitter_addr ON event (emitter_addr)`, | ||||
| 
 | ||||
| 	`CREATE TABLE IF NOT EXISTS event_entry ( | ||||
| 		event_id INTEGER, | ||||
| @ -56,6 +57,8 @@ var ddls = []string{ | ||||
| 		value BLOB NOT NULL | ||||
| 	)`, | ||||
| 
 | ||||
| 	`CREATE INDEX IF NOT EXISTS event_entry_key_index ON event_entry (key)`, | ||||
| 
 | ||||
| 	// metadata containing version of schema
 | ||||
| 	`CREATE TABLE IF NOT EXISTS _meta ( | ||||
|     	version UINT64 NOT NULL UNIQUE | ||||
| @ -63,6 +66,7 @@ var ddls = []string{ | ||||
| 
 | ||||
| 	`INSERT OR IGNORE INTO _meta (version) VALUES (1)`, | ||||
| 	`INSERT OR IGNORE INTO _meta (version) VALUES (2)`, | ||||
| 	`INSERT OR IGNORE INTO _meta (version) VALUES (3)`, | ||||
| } | ||||
| 
 | ||||
| var ( | ||||
| @ -70,7 +74,7 @@ var ( | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	schemaVersion = 2 | ||||
| 	schemaVersion = 3 | ||||
| 
 | ||||
| 	eventExists          = `SELECT MAX(id) FROM event WHERE height=? AND tipset_key=? AND tipset_key_cid=? AND emitter_addr=? AND event_index=? AND message_cid=? AND message_index=?` | ||||
| 	insertEvent          = `INSERT OR IGNORE INTO event(height, tipset_key, tipset_key_cid, emitter_addr, event_index, message_cid, message_index, reverted) VALUES(?, ?, ?, ?, ?, ?, ?, ?)` | ||||
| @ -321,6 +325,22 @@ func NewEventIndex(ctx context.Context, path string, chainStore *store.ChainStor | ||||
| 			version = 2 | ||||
| 		} | ||||
| 
 | ||||
| 		if version == 2 { | ||||
| 			log.Infof("upgrading event index from version 1 to version 2") | ||||
| 
 | ||||
| 			// to upgrade to version 3 we only need to create an index on the event_entry.key column
 | ||||
| 			// and on the event.emitter_addr column
 | ||||
| 			// which means we can just reapply the schema (it will not have any effect on existing data)
 | ||||
| 			for _, ddl := range ddls { | ||||
| 				if _, err := db.Exec(ddl); err != nil { | ||||
| 					_ = db.Close() | ||||
| 					return nil, xerrors.Errorf("could not upgrade index to version 3, exec ddl %q: %w", ddl, err) | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			version = 3 | ||||
| 		} | ||||
| 
 | ||||
| 		if version != schemaVersion { | ||||
| 			_ = db.Close() | ||||
| 			return nil, xerrors.Errorf("invalid database version: got %d, expected %d", version, schemaVersion) | ||||
| @ -481,7 +501,7 @@ func (ei *EventIndex) CollectEvents(ctx context.Context, te *TipSetEvents, rever | ||||
| } | ||||
| 
 | ||||
| // PrefillFilter fills a filter's collection of events from the historic index
 | ||||
| func (ei *EventIndex) PrefillFilter(ctx context.Context, f *EventFilter) error { | ||||
| func (ei *EventIndex) PrefillFilter(ctx context.Context, f *EventFilter, excludeReverted bool) error { | ||||
| 	clauses := []string{} | ||||
| 	values := []any{} | ||||
| 	joins := []string{} | ||||
| @ -500,6 +520,11 @@ func (ei *EventIndex) PrefillFilter(ctx context.Context, f *EventFilter) error { | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if excludeReverted { | ||||
| 		clauses = append(clauses, "event.reverted=?") | ||||
| 		values = append(values, false) | ||||
| 	} | ||||
| 
 | ||||
| 	if len(f.addresses) > 0 { | ||||
| 		subclauses := []string{} | ||||
| 		for _, addr := range f.addresses { | ||||
|  | ||||
| @ -272,7 +272,7 @@ func TestEventIndexPrefillFilter(t *testing.T) { | ||||
| 	for _, tc := range testCases { | ||||
| 		tc := tc // appease lint
 | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			if err := ei.PrefillFilter(context.Background(), tc.filter); err != nil { | ||||
| 			if err := ei.PrefillFilter(context.Background(), tc.filter, false); err != nil { | ||||
| 				require.NoError(t, err, "prefill filter events") | ||||
| 			} | ||||
| 
 | ||||
| @ -281,3 +281,619 @@ func TestEventIndexPrefillFilter(t *testing.T) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func TestEventIndexPrefillFilterExcludeReverted(t *testing.T) { | ||||
| 	rng := pseudo.New(pseudo.NewSource(299792458)) | ||||
| 	a1 := randomF4Addr(t, rng) | ||||
| 	a2 := randomF4Addr(t, rng) | ||||
| 	a3 := randomF4Addr(t, rng) | ||||
| 
 | ||||
| 	a1ID := abi.ActorID(1) | ||||
| 	a2ID := abi.ActorID(2) | ||||
| 
 | ||||
| 	addrMap := addressMap{} | ||||
| 	addrMap.add(a1ID, a1) | ||||
| 	addrMap.add(a2ID, a2) | ||||
| 
 | ||||
| 	ev1 := fakeEvent( | ||||
| 		a1ID, | ||||
| 		[]kv{ | ||||
| 			{k: "type", v: []byte("approval")}, | ||||
| 			{k: "signer", v: []byte("addr1")}, | ||||
| 		}, | ||||
| 		[]kv{ | ||||
| 			{k: "amount", v: []byte("2988181")}, | ||||
| 		}, | ||||
| 	) | ||||
| 	ev2 := fakeEvent( | ||||
| 		a2ID, | ||||
| 		[]kv{ | ||||
| 			{k: "type", v: []byte("approval")}, | ||||
| 			{k: "signer", v: []byte("addr2")}, | ||||
| 		}, | ||||
| 		[]kv{ | ||||
| 			{k: "amount", v: []byte("2988182")}, | ||||
| 		}, | ||||
| 	) | ||||
| 
 | ||||
| 	st := newStore() | ||||
| 	events := []*types.Event{ev1} | ||||
| 	revertedEvents := []*types.Event{ev2} | ||||
| 	em := executedMessage{ | ||||
| 		msg: fakeMessage(randomF4Addr(t, rng), randomF4Addr(t, rng)), | ||||
| 		rct: fakeReceipt(t, rng, st, events), | ||||
| 		evs: events, | ||||
| 	} | ||||
| 	revertedEm := executedMessage{ | ||||
| 		msg: fakeMessage(randomF4Addr(t, rng), randomF4Addr(t, rng)), | ||||
| 		rct: fakeReceipt(t, rng, st, revertedEvents), | ||||
| 		evs: revertedEvents, | ||||
| 	} | ||||
| 
 | ||||
| 	events14000 := buildTipSetEvents(t, rng, 14000, em) | ||||
| 	revertedEvents14000 := buildTipSetEvents(t, rng, 14000, revertedEm) | ||||
| 	cid14000, err := events14000.msgTs.Key().Cid() | ||||
| 	require.NoError(t, err, "tipset cid") | ||||
| 	reveredCID14000, err := revertedEvents14000.msgTs.Key().Cid() | ||||
| 	require.NoError(t, err, "tipset cid") | ||||
| 
 | ||||
| 	noCollectedEvents := []*CollectedEvent{} | ||||
| 	oneCollectedEvent := []*CollectedEvent{ | ||||
| 		{ | ||||
| 			Entries:     ev1.Entries, | ||||
| 			EmitterAddr: a1, | ||||
| 			EventIdx:    0, | ||||
| 			Reverted:    false, | ||||
| 			Height:      14000, | ||||
| 			TipSetKey:   events14000.msgTs.Key(), | ||||
| 			MsgIdx:      0, | ||||
| 			MsgCid:      em.msg.Cid(), | ||||
| 		}, | ||||
| 	} | ||||
| 	twoCollectedEvent := []*CollectedEvent{ | ||||
| 		{ | ||||
| 			Entries:     ev1.Entries, | ||||
| 			EmitterAddr: a1, | ||||
| 			EventIdx:    0, | ||||
| 			Reverted:    false, | ||||
| 			Height:      14000, | ||||
| 			TipSetKey:   events14000.msgTs.Key(), | ||||
| 			MsgIdx:      0, | ||||
| 			MsgCid:      em.msg.Cid(), | ||||
| 		}, | ||||
| 		{ | ||||
| 			Entries:     ev2.Entries, | ||||
| 			EmitterAddr: a2, | ||||
| 			EventIdx:    0, | ||||
| 			Reverted:    true, | ||||
| 			Height:      14000, | ||||
| 			TipSetKey:   revertedEvents14000.msgTs.Key(), | ||||
| 			MsgIdx:      0, | ||||
| 			MsgCid:      revertedEm.msg.Cid(), | ||||
| 		}, | ||||
| 	} | ||||
| 	oneCollectedRevertedEvent := []*CollectedEvent{ | ||||
| 		{ | ||||
| 			Entries:     ev2.Entries, | ||||
| 			EmitterAddr: a2, | ||||
| 			EventIdx:    0, | ||||
| 			Reverted:    true, | ||||
| 			Height:      14000, | ||||
| 			TipSetKey:   revertedEvents14000.msgTs.Key(), | ||||
| 			MsgIdx:      0, | ||||
| 			MsgCid:      revertedEm.msg.Cid(), | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	workDir, err := os.MkdirTemp("", "lotusevents") | ||||
| 	require.NoError(t, err, "create temporary work directory") | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		_ = os.RemoveAll(workDir) | ||||
| 	}() | ||||
| 	t.Logf("using work dir %q", workDir) | ||||
| 
 | ||||
| 	dbPath := filepath.Join(workDir, "actorevents.db") | ||||
| 
 | ||||
| 	ei, err := NewEventIndex(context.Background(), dbPath, nil) | ||||
| 	require.NoError(t, err, "create event index") | ||||
| 	if err := ei.CollectEvents(context.Background(), revertedEvents14000, false, addrMap.ResolveAddress); err != nil { | ||||
| 		require.NoError(t, err, "collect reverted events") | ||||
| 	} | ||||
| 	if err := ei.CollectEvents(context.Background(), revertedEvents14000, true, addrMap.ResolveAddress); err != nil { | ||||
| 		require.NoError(t, err, "revert reverted events") | ||||
| 	} | ||||
| 	if err := ei.CollectEvents(context.Background(), events14000, false, addrMap.ResolveAddress); err != nil { | ||||
| 		require.NoError(t, err, "collect events") | ||||
| 	} | ||||
| 
 | ||||
| 	inclusiveTestCases := []struct { | ||||
| 		name   string | ||||
| 		filter *EventFilter | ||||
| 		te     *TipSetEvents | ||||
| 		want   []*CollectedEvent | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "nomatch tipset min height", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: 14001, | ||||
| 				maxHeight: -1, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch tipset max height", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: 13999, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match tipset min height", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: 14000, | ||||
| 				maxHeight: -1, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: twoCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match tipset cid", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				tipsetCid: cid14000, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match tipset cid", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				tipsetCid: reveredCID14000, | ||||
| 			}, | ||||
| 			te:   revertedEvents14000, | ||||
| 			want: oneCollectedRevertedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch address", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				addresses: []address.Address{a3}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match address 2", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				addresses: []address.Address{a2}, | ||||
| 			}, | ||||
| 			te:   revertedEvents14000, | ||||
| 			want: oneCollectedRevertedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match address 1", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				addresses: []address.Address{a1}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: twoCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry with alternate values", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("cancel"), | ||||
| 						[]byte("propose"), | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: twoCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry by missing value", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("cancel"), | ||||
| 						[]byte("propose"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry by missing key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"method": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry with multiple keys", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"signer": { | ||||
| 						[]byte("addr1"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry with multiple keys", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"signer": { | ||||
| 						[]byte("addr2"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   revertedEvents14000, | ||||
| 			want: oneCollectedRevertedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one mismatching key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"approver": { | ||||
| 						[]byte("addr1"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one mismatching value", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"signer": { | ||||
| 						[]byte("addr3"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one unindexed key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"amount": { | ||||
| 						[]byte("2988181"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one unindexed key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"amount": { | ||||
| 						[]byte("2988182"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	exclusiveTestCases := []struct { | ||||
| 		name   string | ||||
| 		filter *EventFilter | ||||
| 		te     *TipSetEvents | ||||
| 		want   []*CollectedEvent | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "nomatch tipset min height", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: 14001, | ||||
| 				maxHeight: -1, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch tipset max height", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: 13999, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match tipset min height", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: 14000, | ||||
| 				maxHeight: -1, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match tipset cid", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				tipsetCid: cid14000, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match tipset cid but reverted", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				tipsetCid: reveredCID14000, | ||||
| 			}, | ||||
| 			te:   revertedEvents14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch address", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				addresses: []address.Address{a3}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch address 2 but reverted", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				addresses: []address.Address{a2}, | ||||
| 			}, | ||||
| 			te:   revertedEvents14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match address", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				addresses: []address.Address{a1}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry with alternate values", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("cancel"), | ||||
| 						[]byte("propose"), | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry by missing value", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("cancel"), | ||||
| 						[]byte("propose"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry by missing key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"method": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "match one entry with multiple keys", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"signer": { | ||||
| 						[]byte("addr1"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: oneCollectedEvent, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one mismatching key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"approver": { | ||||
| 						[]byte("addr1"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with matching reverted value", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"signer": { | ||||
| 						[]byte("addr2"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one mismatching value", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"type": { | ||||
| 						[]byte("approval"), | ||||
| 					}, | ||||
| 					"signer": { | ||||
| 						[]byte("addr3"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "nomatch one entry with one unindexed key", | ||||
| 			filter: &EventFilter{ | ||||
| 				minHeight: -1, | ||||
| 				maxHeight: -1, | ||||
| 				keys: map[string][][]byte{ | ||||
| 					"amount": { | ||||
| 						[]byte("2988181"), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			te:   events14000, | ||||
| 			want: noCollectedEvents, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	for _, tc := range inclusiveTestCases { | ||||
| 		tc := tc // appease lint
 | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			if err := ei.PrefillFilter(context.Background(), tc.filter, false); err != nil { | ||||
| 				require.NoError(t, err, "prefill filter events") | ||||
| 			} | ||||
| 
 | ||||
| 			coll := tc.filter.TakeCollectedEvents(context.Background()) | ||||
| 			require.ElementsMatch(t, coll, tc.want, tc.name) | ||||
| 		}) | ||||
| 	} | ||||
| 
 | ||||
| 	for _, tc := range exclusiveTestCases { | ||||
| 		tc := tc // appease lint
 | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			if err := ei.PrefillFilter(context.Background(), tc.filter, true); err != nil { | ||||
| 				require.NoError(t, err, "prefill filter events") | ||||
| 			} | ||||
| 
 | ||||
| 			coll := tc.filter.TakeCollectedEvents(context.Background()) | ||||
| 			require.ElementsMatch(t, coll, tc.want, tc.name) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -799,6 +799,45 @@ func GetContractEthAddressFromCode(sender EthAddress, salt [32]byte, initcode [] | ||||
| 	return ethAddr, nil | ||||
| } | ||||
| 
 | ||||
| // EthEstimateGasParams handles raw jsonrpc params for eth_estimateGas
 | ||||
| type EthEstimateGasParams struct { | ||||
| 	Tx       EthCall | ||||
| 	BlkParam *EthBlockNumberOrHash | ||||
| } | ||||
| 
 | ||||
| func (e *EthEstimateGasParams) UnmarshalJSON(b []byte) error { | ||||
| 	var params []json.RawMessage | ||||
| 	err := json.Unmarshal(b, ¶ms) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	switch len(params) { | ||||
| 	case 2: | ||||
| 		err = json.Unmarshal(params[1], &e.BlkParam) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		fallthrough | ||||
| 	case 1: | ||||
| 		err = json.Unmarshal(params[0], &e.Tx) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	default: | ||||
| 		return xerrors.Errorf("expected 1 or 2 params, got %d", len(params)) | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (e EthEstimateGasParams) MarshalJSON() ([]byte, error) { | ||||
| 	if e.BlkParam != nil { | ||||
| 		return json.Marshal([]interface{}{e.Tx, e.BlkParam}) | ||||
| 	} | ||||
| 	return json.Marshal([]interface{}{e.Tx}) | ||||
| } | ||||
| 
 | ||||
| // EthFeeHistoryParams handles raw jsonrpc params for eth_feeHistory
 | ||||
| type EthFeeHistoryParams struct { | ||||
| 	BlkCount          EthUint64 | ||||
|  | ||||
| @ -2402,14 +2402,7 @@ Perms: read | ||||
| Inputs: | ||||
| ```json | ||||
| [ | ||||
|   { | ||||
|     "from": "0x5cbeecf99d3fdb3f25e309cc264f240bb0664031", | ||||
|     "to": "0x5cbeecf99d3fdb3f25e309cc264f240bb0664031", | ||||
|     "gas": "0x5", | ||||
|     "gasPrice": "0x0", | ||||
|     "value": "0x0", | ||||
|     "data": "0x07" | ||||
|   } | ||||
|   "Bw==" | ||||
| ] | ||||
| ``` | ||||
| 
 | ||||
|  | ||||
| @ -131,7 +131,7 @@ type TargetAPI interface { | ||||
| 	EthGasPrice(ctx context.Context) (ethtypes.EthBigInt, error) | ||||
| 	EthFeeHistory(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthFeeHistory, error) | ||||
| 	EthMaxPriorityFeePerGas(ctx context.Context) (ethtypes.EthBigInt, error) | ||||
| 	EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error) | ||||
| 	EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error) | ||||
| 	EthCall(ctx context.Context, tx ethtypes.EthCall, blkParam ethtypes.EthBlockNumberOrHash) (ethtypes.EthBytes, error) | ||||
| 	EthSendRawTransaction(ctx context.Context, rawTx ethtypes.EthBytes) (ethtypes.EthHash, error) | ||||
| 	EthGetLogs(ctx context.Context, filter *ethtypes.EthFilterSpec) (*ethtypes.EthFilterResult, error) | ||||
|  | ||||
| @ -353,13 +353,19 @@ func (gw *Node) EthMaxPriorityFeePerGas(ctx context.Context) (ethtypes.EthBigInt | ||||
| 	return gw.target.EthMaxPriorityFeePerGas(ctx) | ||||
| } | ||||
| 
 | ||||
| func (gw *Node) EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (gw *Node) EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	// validate params
 | ||||
| 	_, err := jsonrpc.DecodeParams[ethtypes.EthEstimateGasParams](p) | ||||
| 	if err != nil { | ||||
| 		return ethtypes.EthUint64(0), xerrors.Errorf("decoding params: %w", err) | ||||
| 	} | ||||
| 
 | ||||
| 	if err := gw.limit(ctx, stateRateLimitTokens); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 
 | ||||
| 	// todo limit gas? to what?
 | ||||
| 	return gw.target.EthEstimateGas(ctx, tx) | ||||
| 	return gw.target.EthEstimateGas(ctx, p) | ||||
| } | ||||
| 
 | ||||
| func (gw *Node) EthCall(ctx context.Context, tx ethtypes.EthCall, blkParam ethtypes.EthBlockNumberOrHash) (ethtypes.EthBytes, error) { | ||||
|  | ||||
| @ -3,6 +3,7 @@ package itests | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/hex" | ||||
| 	"encoding/json" | ||||
| 	"os" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @ -272,10 +273,13 @@ func TestEthAccountAbstractionFailsFromEvmActor(t *testing.T) { | ||||
| 	contract, err := hex.DecodeString(string(contractHex)) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		Data: contract, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
|  | ||||
| @ -35,7 +35,7 @@ type ethAPIRaw struct { | ||||
| 	EthBlockNumber                         func(context.Context) (json.RawMessage, error) | ||||
| 	EthCall                                func(context.Context, ethtypes.EthCall, ethtypes.EthBlockNumberOrHash) (json.RawMessage, error) | ||||
| 	EthChainId                             func(context.Context) (json.RawMessage, error) | ||||
| 	EthEstimateGas                         func(context.Context, ethtypes.EthCall) (json.RawMessage, error) | ||||
| 	EthEstimateGas                         func(context.Context, jsonrpc.RawParams) (json.RawMessage, error) | ||||
| 	EthFeeHistory                          func(context.Context, ethtypes.EthUint64, string, []float64) (json.RawMessage, error) | ||||
| 	EthGasPrice                            func(context.Context) (json.RawMessage, error) | ||||
| 	EthGetBalance                          func(context.Context, ethtypes.EthAddress, ethtypes.EthBlockNumberOrHash) (json.RawMessage, error) | ||||
| @ -182,10 +182,13 @@ func TestEthOpenRPCConformance(t *testing.T) { | ||||
| 		{ | ||||
| 			method: "eth_estimateGas", | ||||
| 			call: func(a *ethAPIRaw) (json.RawMessage, error) { | ||||
| 				return ethapi.EthEstimateGas(context.Background(), ethtypes.EthCall{ | ||||
| 				gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 					From: &senderEthAddr, | ||||
| 					Data: contractBin, | ||||
| 				}) | ||||
| 				}}) | ||||
| 				require.NoError(t, err) | ||||
| 
 | ||||
| 				return ethapi.EthEstimateGas(ctx, gasParams) | ||||
| 			}, | ||||
| 		}, | ||||
| 
 | ||||
| @ -448,10 +451,13 @@ func TestEthOpenRPCConformance(t *testing.T) { | ||||
| } | ||||
| 
 | ||||
| func createRawSignedEthTx(ctx context.Context, t *testing.T, client *kit.TestFullNode, senderEthAddr ethtypes.EthAddress, receiverEthAddr ethtypes.EthAddress, senderKey *key.Key, contractBin []byte) []byte { | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: &senderEthAddr, | ||||
| 		Data: contractBin, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
|  | ||||
| @ -60,10 +60,13 @@ func TestDeployment(t *testing.T) { | ||||
| 	// verify the deployer address is an Placeholder.
 | ||||
| 	client.AssertActorType(ctx, deployer, manifest.PlaceholderKey) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		Data: contract, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
|  | ||||
| @ -3,6 +3,7 @@ package itests | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/hex" | ||||
| 	"encoding/json" | ||||
| 	"os" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @ -47,10 +48,13 @@ func TestTransactionHashLookup(t *testing.T) { | ||||
| 	// send some funds to the f410 address
 | ||||
| 	kit.SendFunds(ctx, t, client, deployer, types.FromFil(10)) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		Data: contract, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
| @ -313,10 +317,13 @@ func TestEthGetMessageCidByTransactionHashEthTx(t *testing.T) { | ||||
| 	// send some funds to the f410 address
 | ||||
| 	kit.SendFunds(ctx, t, client, deployer, types.FromFil(10)) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		Data: contract, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
|  | ||||
| @ -3,6 +3,7 @@ package itests | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/hex" | ||||
| 	"encoding/json" | ||||
| 	"os" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @ -42,12 +43,19 @@ func TestValueTransferValidSignature(t *testing.T) { | ||||
| 
 | ||||
| 	kit.SendFunds(ctx, t, client, deployer, types.FromFil(1000)) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		Data: contract, | ||||
| 	blkParam := ethtypes.NewEthBlockNumberOrHashFromPredefined("latest") | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{ | ||||
| 		Tx: ethtypes.EthCall{ | ||||
| 			From: ðAddr, | ||||
| 			Data: contract, | ||||
| 		}, | ||||
| 		BlkParam: &blkParam, | ||||
| 	}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| @ -229,11 +237,14 @@ func TestContractInvocation(t *testing.T) { | ||||
| 	params, err := hex.DecodeString("f8b2cb4f000000000000000000000000ff00000000000000000000000000000000000064") | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		To:   &contractAddr, | ||||
| 		Data: params, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
| @ -348,10 +359,15 @@ func TestGetBlockByNumber(t *testing.T) { | ||||
| } | ||||
| 
 | ||||
| func deployContractTx(ctx context.Context, client *kit.TestFullNode, ethAddr ethtypes.EthAddress, contract []byte) (*ethtypes.EthTxArgs, error) { | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From: ðAddr, | ||||
| 		Data: contract, | ||||
| 	}) | ||||
| 	}}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| @ -6,6 +6,7 @@ import ( | ||||
| 	"crypto/rand" | ||||
| 	"encoding/binary" | ||||
| 	"encoding/hex" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @ -657,11 +658,15 @@ func TestFEVMRecursiveActorCallEstimate(t *testing.T) { | ||||
| 			t.Logf("running with %d recursive calls", r) | ||||
| 
 | ||||
| 			params := makeParams(r) | ||||
| 			gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 
 | ||||
| 			gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 				From: ðAddr, | ||||
| 				To:   &contractAddr, | ||||
| 				Data: params, | ||||
| 			}) | ||||
| 			}}) | ||||
| 			require.NoError(t, err) | ||||
| 
 | ||||
| 			gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 			require.NoError(t, err) | ||||
| 			require.LessOrEqual(t, int64(gaslimit), build.BlockGasLimit) | ||||
| 
 | ||||
| @ -816,11 +821,14 @@ func TestFEVMBareTransferTriggersSmartContractLogic(t *testing.T) { | ||||
| 	contractEth, err := ethtypes.EthAddressFromFilecoinAddress(contractAddr) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 	gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 		From:  &accntEth, | ||||
| 		To:    &contractEth, | ||||
| 		Value: ethtypes.EthBigInt(big.NewInt(100)), | ||||
| 	}) | ||||
| 	}}) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	gaslimit, err := client.EthEstimateGas(ctx, gasParams) | ||||
| 	require.NoError(t, err) | ||||
| 
 | ||||
| 	maxPriorityFeePerGas, err := client.EthMaxPriorityFeePerGas(ctx) | ||||
| @ -1034,10 +1042,13 @@ func TestFEVMErrorParsing(t *testing.T) { | ||||
| 				require.ErrorContains(t, err, expected) | ||||
| 			}) | ||||
| 			t.Run("EthEstimateGas", func(t *testing.T) { | ||||
| 				_, err := e.EthEstimateGas(ctx, ethtypes.EthCall{ | ||||
| 				gasParams, err := json.Marshal(ethtypes.EthEstimateGasParams{Tx: ethtypes.EthCall{ | ||||
| 					To:   &contractAddrEth, | ||||
| 					Data: entryPoint, | ||||
| 				}) | ||||
| 				}}) | ||||
| 				require.NoError(t, err) | ||||
| 
 | ||||
| 				_, err = e.EthEstimateGas(ctx, gasParams) | ||||
| 				require.ErrorContains(t, err, expected) | ||||
| 			}) | ||||
| 		}) | ||||
|  | ||||
| @ -122,7 +122,7 @@ func (e *EthModuleDummy) EthGasPrice(ctx context.Context) (ethtypes.EthBigInt, e | ||||
| 	return ethtypes.EthBigIntZero, ErrModuleDisabled | ||||
| } | ||||
| 
 | ||||
| func (e *EthModuleDummy) EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| func (e *EthModuleDummy) EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	return 0, ErrModuleDisabled | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -66,7 +66,7 @@ type EthModuleAPI interface { | ||||
| 	NetListening(ctx context.Context) (bool, error) | ||||
| 	EthProtocolVersion(ctx context.Context) (ethtypes.EthUint64, error) | ||||
| 	EthGasPrice(ctx context.Context) (ethtypes.EthBigInt, error) | ||||
| 	EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error) | ||||
| 	EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error) | ||||
| 	EthCall(ctx context.Context, tx ethtypes.EthCall, blkParam ethtypes.EthBlockNumberOrHash) (ethtypes.EthBytes, error) | ||||
| 	EthMaxPriorityFeePerGas(ctx context.Context) (ethtypes.EthBigInt, error) | ||||
| 	EthSendRawTransaction(ctx context.Context, rawTx ethtypes.EthBytes) (ethtypes.EthHash, error) | ||||
| @ -999,8 +999,13 @@ func (a *EthModule) applyMessage(ctx context.Context, msg *types.Message, tsk ty | ||||
| 	return res, nil | ||||
| } | ||||
| 
 | ||||
| func (a *EthModule) EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (ethtypes.EthUint64, error) { | ||||
| 	msg, err := ethCallToFilecoinMessage(ctx, tx) | ||||
| func (a *EthModule) EthEstimateGas(ctx context.Context, p jsonrpc.RawParams) (ethtypes.EthUint64, error) { | ||||
| 	params, err := jsonrpc.DecodeParams[ethtypes.EthEstimateGasParams](p) | ||||
| 	if err != nil { | ||||
| 		return ethtypes.EthUint64(0), xerrors.Errorf("decoding params: %w", err) | ||||
| 	} | ||||
| 
 | ||||
| 	msg, err := ethCallToFilecoinMessage(ctx, params.Tx) | ||||
| 	if err != nil { | ||||
| 		return ethtypes.EthUint64(0), err | ||||
| 	} | ||||
| @ -1009,7 +1014,16 @@ func (a *EthModule) EthEstimateGas(ctx context.Context, tx ethtypes.EthCall) (et | ||||
| 	// gas estimation actually run.
 | ||||
| 	msg.GasLimit = 0 | ||||
| 
 | ||||
| 	ts := a.Chain.GetHeaviestTipSet() | ||||
| 	var ts *types.TipSet | ||||
| 	if params.BlkParam == nil { | ||||
| 		ts = a.Chain.GetHeaviestTipSet() | ||||
| 	} else { | ||||
| 		ts, err = getTipsetByEthBlockNumberOrHash(ctx, a.Chain, *params.BlkParam) | ||||
| 		if err != nil { | ||||
| 			return ethtypes.EthUint64(0), xerrors.Errorf("failed to process block param: %v; %w", params.BlkParam, err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	gassedMsg, err := a.GasAPI.GasEstimateMessageGas(ctx, msg, nil, ts.Key()) | ||||
| 	if err != nil { | ||||
| 		// On failure, GasEstimateMessageGas doesn't actually return the invocation result,
 | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user