diff --git a/fn/iter.go b/fn/iter.go index c82d41639..45e0fdc67 100644 --- a/fn/iter.go +++ b/fn/iter.go @@ -28,9 +28,9 @@ func ForEach[T any](items []T, f func(T)) { // ForEachMapItem is a generic implementation of a for-each (map with side // effects). This can be used to ensure that any normal for-loop don't run into // bugs due to loop variable scoping. -func ForEachMapItem[T any, K comparable](items map[K]T, f func(T)) { +func ForEachMapItem[T any, K comparable](items map[K]T, f func(K, T)) { for i := range items { - f(items[i]) + f(i, items[i]) } } diff --git a/tapgarden/batch.go b/tapgarden/batch.go index 6d21a7621..edf1fb633 100644 --- a/tapgarden/batch.go +++ b/tapgarden/batch.go @@ -188,3 +188,7 @@ func (m *MintingBatch) TapSibling() []byte { func (m *MintingBatch) UpdateTapSibling(sibling *chainhash.Hash) { m.tapSibling = sibling } + +func (m *MintingBatch) IsFunded() bool { + return m.GenesisPacket != nil +} diff --git a/tapgarden/caretaker.go b/tapgarden/caretaker.go index d73ba11d8..f19efeefe 100644 --- a/tapgarden/caretaker.go +++ b/tapgarden/caretaker.go @@ -402,15 +402,71 @@ func (b *BatchCaretaker) seedlingsToAssetSprouts(ctx context.Context, newAssets := make([]*asset.Asset, 0, len(b.cfg.Batch.Seedlings)) - // Seedlings that anchor a group may be referenced by other seedlings, - // and therefore need to be mapped to sprouts first so that we derive - // the initial tweaked group key early. - orderedSeedlings := SortSeedlings(maps.Values(b.cfg.Batch.Seedlings)) - newGroups := make(map[string]*asset.AssetGroup, len(orderedSeedlings)) + // separate grouped assets from ungrouped + groupedSeedlings, ungroupedSeedlings := filterSeedlingsWithGroup( + b.cfg.Batch.Seedlings, + ) + groupedSeedlingCount := len(groupedSeedlings) + + // load seedling asset groups and check for correct group count + seedlingGroups, err := b.cfg.Log.FetchSeedlingGroups( + ctx, genesisPoint, assetOutputIndex, + maps.Values(groupedSeedlings), + ) + if err != nil { + return nil, err + } + seedlingGroupCount := len(seedlingGroups) + + if groupedSeedlingCount != seedlingGroupCount { + return nil, fmt.Errorf("wrong number of grouped assets and "+ + "asset groups: %d, %d", groupedSeedlingCount, + seedlingGroupCount) + } + + for i := range seedlingGroups { + // check that asset group has a witness, and that the group + // has a matching seedling + seedlingGroup := seedlingGroups[i] + if len(seedlingGroup.GroupKey.Witness) == 0 { + return nil, fmt.Errorf("not all seedling groups have " + + "witnesses") + } + + seedling, ok := groupedSeedlings[seedlingGroup.Tag] + if !ok { + groupTweakedKey := seedlingGroup.GroupKey.GroupPubKey. + SerializeCompressed() + return nil, fmt.Errorf("no seedling with tag matching "+ + "group: %v, %x", seedlingGroup.Tag, + groupTweakedKey) + } + + // build assets for grouped seedlings + var amount uint64 + switch seedling.AssetType { + case asset.Normal: + amount = seedling.Amount + case asset.Collectible: + amount = 1 + } + + newAsset, err := asset.New( + *seedlingGroup.Genesis, amount, 0, 0, + *seedling.ScriptKey, seedlingGroup.GroupKey, + asset.WithAssetVersion(seedling.AssetVersion), + ) + if err != nil { + return nil, fmt.Errorf("unable to create new asset: %w", + err) + } - for _, seedlingName := range orderedSeedlings { - seedling := b.cfg.Batch.Seedlings[seedlingName] + newAssets = append(newAssets, newAsset) + } + // build assets for ungrouped seedlings + for seedlingName := range ungroupedSeedlings { + seedling := ungroupedSeedlings[seedlingName] assetGen := asset.Genesis{ FirstPrevOut: genesisPoint, Tag: seedling.AssetName, @@ -429,15 +485,8 @@ func (b *BatchCaretaker) seedlingsToAssetSprouts(ctx context.Context, return nil, fmt.Errorf("unable to obtain script key") } - var ( - amount uint64 - groupInfo *asset.AssetGroup - protoAsset *asset.Asset - sproutGroupKey *asset.GroupKey - err error - ) - // Determine the amount for the actual asset. + var amount uint64 switch seedling.AssetType { case asset.Normal: amount = seedling.Amount @@ -445,106 +494,10 @@ func (b *BatchCaretaker) seedlingsToAssetSprouts(ctx context.Context, amount = 1 } - // If the seedling has a group key specified, - // that group key was validated earlier. We need to - // sign the new genesis with that group key. - if seedling.HasGroupKey() { - groupInfo = seedling.GroupInfo - } - - // If the seedling has a group anchor specified, that anchor - // was validated earlier and the corresponding group has already - // been created. We need to look up the group key and sign - // the asset genesis with that key. - if seedling.GroupAnchor != nil { - groupInfo = newGroups[*seedling.GroupAnchor] - } - - // If a group witness needs to be produced, then we will need a - // partially filled asset as part of the signing process. - if groupInfo != nil || seedling.EnableEmission { - protoAsset, err = asset.New( - assetGen, amount, 0, 0, *seedling.ScriptKey, - nil, - asset.WithAssetVersion(seedling.AssetVersion), - ) - if err != nil { - return nil, fmt.Errorf("unable to create "+ - "asset for group key signing: %w", err) - } - } - - if groupInfo != nil { - groupReq, err := asset.NewGroupKeyRequest( - groupInfo.GroupKey.RawKey, *groupInfo.Genesis, - protoAsset, nil, - ) - if err != nil { - return nil, fmt.Errorf("unable to request "+ - "asset group membership: %w", err) - } - - genTx, err := groupReq.BuildGenesisTx( - b.cfg.GenTxBuilder, - ) - if err != nil { - return nil, err - } - - sproutGroupKey, err = asset.DeriveGroupKey( - b.cfg.GenSigner, *genTx, *groupReq, - ) - if err != nil { - return nil, fmt.Errorf("unable to tweak group "+ - "key: %w", err) - } - } - - // If emission is enabled without a group key specified, - // then we'll need to generate another public key, - // then use that to derive the key group signature - // along with the tweaked key group. - if seedling.EnableEmission { - if seedling.GroupInternalKey == nil { - return nil, fmt.Errorf("unable to derive " + - "group key") - } - - groupReq, err := asset.NewGroupKeyRequest( - *seedling.GroupInternalKey, assetGen, - protoAsset, nil, - ) - if err != nil { - return nil, fmt.Errorf("unable to request "+ - "asset group creation: %w", err) - } - - genTx, err := groupReq.BuildGenesisTx( - b.cfg.GenTxBuilder, - ) - if err != nil { - return nil, err - } - - sproutGroupKey, err = asset.DeriveGroupKey( - b.cfg.GenSigner, *genTx, *groupReq, - ) - if err != nil { - return nil, fmt.Errorf("unable to tweak group "+ - "key: %w", err) - } - - newGroups[seedlingName] = &asset.AssetGroup{ - Genesis: &assetGen, - GroupKey: sproutGroupKey, - } - } - // With the necessary keys components assembled, we'll create // the actual asset now. newAsset, err := asset.New( - assetGen, amount, 0, 0, *seedling.ScriptKey, - sproutGroupKey, + assetGen, amount, 0, 0, *seedling.ScriptKey, nil, asset.WithAssetVersion(seedling.AssetVersion), ) if err != nil { @@ -552,15 +505,6 @@ func (b *BatchCaretaker) seedlingsToAssetSprouts(ctx context.Context, err) } - // Verify the group witness if present. - if sproutGroupKey != nil { - err := b.cfg.TxValidator.Execute(newAsset, nil, nil) - if err != nil { - return nil, fmt.Errorf("unable to verify "+ - "asset group witness: %w", err) - } - } - newAssets = append(newAssets, newAsset) } diff --git a/tapgarden/planter.go b/tapgarden/planter.go index 3876438e3..8967e8dcc 100644 --- a/tapgarden/planter.go +++ b/tapgarden/planter.go @@ -1186,23 +1186,24 @@ func (c *ChainPlanter) finalizeBatch(params FinalizeParams) (*BatchCaretaker, } // If the batch already has a funded TX, we can skip funding the batch. - if c.pendingBatch.GenesisPacket == nil { - fundParams := FundParams(params) - + if !c.pendingBatch.IsFunded() { // Fund the batch before starting the caretaker. If funding // fails, we can't start a caretaker for the batch, so we'll // clear the pending batch. The batch will exist on disk for // the user to recreate it if necessary. - err = c.fundBatch(ctx, fundParams) + // TODO(jhb): Don't clear pending batch here + err = c.fundBatch(ctx, FundParams(params)) if err != nil { c.pendingBatch = nil return nil, err } } - // TODO(jhb): move batch sibling handling entirely to fundBatch, remove - // logic around sibling storage - // TODO(jhb): check for batch sealing + // TODO(jhb): check for batch already sealed + err = c.sealBatch(ctx, SealParams{}) + if err != nil { + return nil, err + } // Now that the batch has been frozen on disk, we can update the batch // state to frozen before launching a new caretaker state machine for