diff --git a/.github/workflows/push_docker_image.yml b/.github/workflows/push_docker_image.yml index 90bde4c02..bd0911e71 100644 --- a/.github/workflows/push_docker_image.yml +++ b/.github/workflows/push_docker_image.yml @@ -53,14 +53,16 @@ jobs: docker build . \ -f ${{ matrix.docker.dockerfile }}.amd64 \ -t ${{ matrix.docker.repo }}:git-${{ github.sha }}-amd64 \ - --build-arg COMMIT=git-${{ github.sha }} + --build-arg COMMIT=git-${{ github.sha }} \ + --build-arg TARGETPLATFORM=linux/amd64 docker push ${{ matrix.docker.repo }}:git-${{ github.sha }}-amd64 - name: build-and-push-arm64v8 run: | docker build . \ -f ${{ matrix.docker.dockerfile }}.arm64v8 \ -t ${{ matrix.docker.repo }}:git-${{ github.sha }}-arm64v8 \ - --build-arg COMMIT=git-${{ github.sha }} + --build-arg COMMIT=git-${{ github.sha }} \ + --build-arg TARGETPLATFORM=linux/arm64 docker push ${{ matrix.docker.repo }}:git-${{ github.sha }}-arm64v8 - name: merge-manifest-and-push run: | diff --git a/Dockerfile b/Dockerfile index 6e03a4942..cea86d51d 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,6 +1,7 @@ -FROM mcr.microsoft.com/dotnet/sdk:8.0-jammy AS build-env +FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build-env WORKDIR /app ARG COMMIT +ARG TARGETPLATFORM # Copy csproj and restore as distinct layers COPY ./Lib9c/Lib9c/Lib9c.csproj ./Lib9c/ @@ -16,23 +17,40 @@ RUN dotnet restore NineChronicles.Headless.Executable # Copy everything else and build COPY . ./ -RUN dotnet publish NineChronicles.Headless.Executable/NineChronicles.Headless.Executable.csproj \ +RUN < new ArenaParticipantsWorker(arenaMemoryCache, standaloneContext, headlessConfig.ArenaParticipantsSyncInterval)); - } services.AddSingleton(arenaMemoryCache); }); diff --git a/NineChronicles.Headless.Tests/ArenaParticipantsWorkerTest.cs b/NineChronicles.Headless.Tests/ArenaParticipantsWorkerTest.cs deleted file mode 100644 index f0835bbc7..000000000 --- a/NineChronicles.Headless.Tests/ArenaParticipantsWorkerTest.cs +++ /dev/null @@ -1,207 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using Lib9c.Tests; -using Libplanet.Action.State; -using Libplanet.Crypto; -using Libplanet.Mocks; -using Nekoyume; -using Nekoyume.Action; -using Nekoyume.Model.Arena; -using Nekoyume.Model.EnumType; -using Nekoyume.Model.Item; -using Nekoyume.Model.State; -using Nekoyume.Module; -using Nekoyume.TableData; -using Xunit; -using Random = Libplanet.Extensions.ActionEvaluatorCommonComponents.Random; - -namespace NineChronicles.Headless.Tests; - -public class ArenaParticipantsWorkerTest -{ - private readonly IWorld _world; - private readonly Dictionary _sheets; - - public ArenaParticipantsWorkerTest() - { - _world = new World(MockWorldState.CreateModern()); - _sheets = TableSheetsImporter.ImportSheets(); - } - - [Fact] - public void GetRoundData() - { - var sheetAddress = Addresses.GetSheetAddress(); - var csv = _sheets[nameof(ArenaSheet)]; - var arenaSheet = new ArenaSheet(); - arenaSheet.Set(csv); - var row = arenaSheet.OrderedList.First(); - var expectedRound = row.Round.First(); - var blockIndex = expectedRound.StartBlockIndex; - var state = _world.SetLegacyState(sheetAddress, csv.Serialize()); - var current = ArenaParticipantsWorker.GetRoundData(state, blockIndex); - Assert.Equal(expectedRound.Round, current.Round); - Assert.Equal(expectedRound.ChampionshipId, current.ChampionshipId); - } - - [Fact] - public void GetArenaParticipantsState() - { - var sheetAddress = Addresses.GetSheetAddress(); - var csv = _sheets[nameof(ArenaSheet)]; - var arenaSheet = new ArenaSheet(); - arenaSheet.Set(csv); - var row = arenaSheet.OrderedList.First(); - var currentRoundData = row.Round.First(); - var state = _world.SetLegacyState(sheetAddress, csv.Serialize()); - Assert.Null(ArenaParticipantsWorker.GetArenaParticipantsState(state, currentRoundData)); - - var participantsAddr = ArenaParticipants.DeriveAddress( - currentRoundData.ChampionshipId, - currentRoundData.Round); - var expected = new ArenaParticipants(currentRoundData.ChampionshipId, currentRoundData.Round); - var avatarAddress = new PrivateKey().Address; - expected.AvatarAddresses.Add(avatarAddress); - state = state.SetLegacyState(participantsAddr, expected.Serialize()); - var actual = ArenaParticipantsWorker.GetArenaParticipantsState(state, currentRoundData); - Assert.NotNull(actual); - Assert.Equal(expected.Serialize(), actual.Serialize()); - } - - [Fact] - public void AvatarAddrAndScoresWithRank() - { - var sheetAddress = Addresses.GetSheetAddress(); - var csv = _sheets[nameof(ArenaSheet)]; - var arenaSheet = new ArenaSheet(); - arenaSheet.Set(csv); - var row = arenaSheet.OrderedList.First(); - var currentRoundData = row.Round.First(); - var championshipId = currentRoundData.ChampionshipId; - var round = currentRoundData.Round; - var participantsAddr = ArenaParticipants.DeriveAddress(championshipId, round); - var participants = new ArenaParticipants(championshipId, round); - var avatarAddress = new PrivateKey().Address; - var avatar2Address = new PrivateKey().Address; - participants.AvatarAddresses.Add(avatarAddress); - participants.AvatarAddresses.Add(avatar2Address); - var arenaScore = new ArenaScore(avatarAddress, championshipId, round); - arenaScore.AddScore(10); - var state = _world - .SetLegacyState(sheetAddress, csv.Serialize()) - .SetLegacyState(participantsAddr, participants.Serialize()) - .SetLegacyState(arenaScore.Address, arenaScore.Serialize()); - var actual = - ArenaParticipantsWorker.AvatarAddrAndScoresWithRank(participants.AvatarAddresses, currentRoundData, state); - Assert.Equal(2, actual.Count); - var first = actual.First(); - Assert.Equal(avatarAddress, first.avatarAddr); - Assert.Equal(1010, first.score); - Assert.Equal(1, first.rank); - var second = actual.Last(); - Assert.Equal(avatar2Address, second.avatarAddr); - Assert.Equal(1000, second.score); - Assert.Equal(2, second.rank); - } - - [Fact] - public void GetArenaParticipants() - { - var tableSheets = new TableSheets(_sheets); - var agentAddress = new PrivateKey().Address; - var avatarAddress = Addresses.GetAvatarAddress(agentAddress, 0); - var avatarState = AvatarState.Create( - avatarAddress, - agentAddress, - 0, - tableSheets.GetAvatarSheets(), - new Address(), - "avatar_state" - ); - var avatar2Address = Addresses.GetAvatarAddress(agentAddress, 1); - var avatarState2 = AvatarState.Create( - avatar2Address, - agentAddress, - 0, - tableSheets.GetAvatarSheets(), - new Address(), - "avatar_state2" - ); - - // equipment - var equipmentSheet = tableSheets.EquipmentItemSheet; - var random = new Random(0); - var equipment = - (Equipment)ItemFactory.CreateItem(equipmentSheet.Values.First(r => r.ItemSubType == ItemSubType.Armor), - random); - equipment.equipped = true; - avatarState.inventory.AddItem(equipment); - avatarState2.inventory.AddItem(equipment); - var itemSlotState = new ItemSlotState(BattleType.Arena); - var itemSlotAddress = ItemSlotState.DeriveAddress(avatarAddress, BattleType.Arena); - itemSlotState.UpdateEquipment(new List - { - equipment.ItemId, - }); - - // rune - var runeListSheet = tableSheets.RuneListSheet; - var runeId = runeListSheet.Values.First().Id; - var runeSlotState = new RuneSlotState(BattleType.Arena); - var runeSlotAddress = RuneSlotState.DeriveAddress(avatarAddress, BattleType.Arena); - var runeSlotInfo = new RuneSlotInfo(0, runeId); - runeSlotState.UpdateSlot(new List - { - runeSlotInfo, - }, runeListSheet); - var runeStates = new AllRuneState(runeId); - - // collection - var collectionSheet = tableSheets.CollectionSheet; - var collectionState = new CollectionState(); - collectionState.Ids.Add(collectionSheet.Values.First().Id); - var arenaSheet = tableSheets.ArenaSheet; - var row = arenaSheet.OrderedList.First(); - var currentRoundData = row.Round.First(); - var championshipId = currentRoundData.ChampionshipId; - var round = currentRoundData.Round; - var participantsAddr = ArenaParticipants.DeriveAddress(championshipId, round); - var participants = new ArenaParticipants(championshipId, round); - participants.AvatarAddresses.Add(avatarAddress); - participants.AvatarAddresses.Add(avatar2Address); - var arenaScore = new ArenaScore(avatarAddress, championshipId, round); - arenaScore.AddScore(10); - var state = _world - .SetAvatarState(avatarAddress, avatarState, true, true, true, true) - .SetAvatarState(avatar2Address, avatarState2, true, true, true, true) - .SetLegacyState(itemSlotAddress, itemSlotState.Serialize()) - .SetRuneState(avatarAddress, runeStates) - .SetLegacyState(runeSlotAddress, runeSlotState.Serialize()) - .SetCollectionState(avatar2Address, collectionState) - .SetLegacyState(participantsAddr, participants.Serialize()) - .SetLegacyState(arenaScore.Address, arenaScore.Serialize()); - foreach (var (key, s) in _sheets) - { - state = state.SetLegacyState(Addresses.GetSheetAddress(key), s.Serialize()); - } - - var avatarAddrAndScoresWithRank = - ArenaParticipantsWorker.AvatarAddrAndScoresWithRank(participants.AvatarAddresses, currentRoundData, state); - var actual = - ArenaParticipantsWorker.GetArenaParticipants(state, participants.AvatarAddresses, - avatarAddrAndScoresWithRank); - Assert.Equal(2, actual.Count); - var first = actual.First(); - Assert.Equal(avatarAddress, first.AvatarAddr); - Assert.Equal(1010, first.Score); - Assert.Equal(1, first.Rank); - Assert.Equal(equipment.Id, first.PortraitId); - var second = actual.Last(); - Assert.Equal(avatar2Address, second.AvatarAddr); - Assert.Equal(1000, second.Score); - Assert.Equal(2, second.Rank); - Assert.Equal(GameConfig.DefaultAvatarArmorId, second.PortraitId); - Assert.True(first.Cp < second.Cp); - } -} diff --git a/NineChronicles.Headless.Tests/GraphTypes/States/Models/AvatarStateTypeTest.cs b/NineChronicles.Headless.Tests/GraphTypes/States/Models/AvatarStateTypeTest.cs index 2bda35e0f..b2e54fc67 100644 --- a/NineChronicles.Headless.Tests/GraphTypes/States/Models/AvatarStateTypeTest.cs +++ b/NineChronicles.Headless.Tests/GraphTypes/States/Models/AvatarStateTypeTest.cs @@ -143,9 +143,9 @@ public async Task QueryActionPoint(bool modern, Dictionary expec ["combinationSlots"] = new World(MockWorldState.CreateModern()).GetAllCombinationSlotState(Fixtures.AvatarAddress).Select(x => new Dictionary { ["address"] = x.address.ToString(), - ["unlockBlockIndex"] = x.UnlockBlockIndex, + ["unlockBlockIndex"] = x.WorkCompleteBlockIndex, ["isUnlocked"] = x.IsUnlocked, - ["startBlockIndex"] = x.StartBlockIndex, + ["startBlockIndex"] = x.WorkStartBlockIndex, ["petId"] = x.PetId }).ToArray(), } diff --git a/NineChronicles.Headless/ArenaParticipantsWorker.cs b/NineChronicles.Headless/ArenaParticipantsWorker.cs deleted file mode 100644 index c31d7f792..000000000 --- a/NineChronicles.Headless/ArenaParticipantsWorker.cs +++ /dev/null @@ -1,323 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Diagnostics; -using System.Linq; -using System.Threading; -using System.Threading.Tasks; -using Bencodex.Types; -using Libplanet.Action.State; -using Libplanet.Crypto; -using Microsoft.Extensions.Caching.Memory; -using Microsoft.Extensions.Hosting; -using Nekoyume; -using Nekoyume.Battle; -using Nekoyume.Helper; -using Nekoyume.Model.Arena; -using Nekoyume.Model.EnumType; -using Nekoyume.Model.Stat; -using Nekoyume.Model.State; -using Nekoyume.Module; -using Nekoyume.TableData; -using Nekoyume.TableData.Rune; -using NineChronicles.Headless.GraphTypes; -using Serilog; - -namespace NineChronicles.Headless; - -public class ArenaParticipantsWorker : BackgroundService -{ - private ILogger _logger; - - private StateMemoryCache _cache; - - private StandaloneContext _context; - - private int _interval; - - public ArenaParticipantsWorker(StateMemoryCache memoryCache, StandaloneContext context, int interval) - { - _cache = memoryCache; - _context = context; - _logger = Log.Logger.ForContext(); - _interval = interval; - } - - protected override async Task ExecuteAsync(CancellationToken stoppingToken) - { - try - { - while (!stoppingToken.IsCancellationRequested) - { - await Task.Delay(_interval, stoppingToken); - PrepareArenaParticipants(); - } - } - catch (OperationCanceledException) - { - //pass - _logger.Information("[ArenaParticipantsWorker]Cancel ArenaParticipantsWorker"); - } - catch (Exception e) - { - _logger.Error(e, "[ArenaParticipantsWorker]Stopping ArenaParticipantsWorker"); - await StopAsync(stoppingToken); - } - } - - /// - /// Retrieves the state of arena participants from the given world state and current round data. - /// - /// The world state. - /// The current round data. - /// The arena participants state, or null if not found. - public static ArenaParticipants? GetArenaParticipantsState(IWorldState worldState, ArenaSheet.RoundData currentRoundData) - { - var participantsAddr = ArenaParticipants.DeriveAddress( - currentRoundData.ChampionshipId, - currentRoundData.Round); - var participants = worldState.GetLegacyState(participantsAddr) is List participantsList - ? new ArenaParticipants(participantsList) - : null; - return participants; - } - - /// - /// Gets the round data from the specified world state and block index. - /// - /// The world state containing the arena sheet. - /// The block index for which to retrieve the round data. - /// The round data for the specified block index. - public static ArenaSheet.RoundData GetRoundData(IWorldState worldState, long blockIndex) - { - return worldState.GetSheet().GetRoundByBlockIndex(blockIndex); - } - - /// - /// Retrieves the avatar addresses and scores with ranks for a given list of avatar addresses, current round data, and world state. - /// - /// The list of avatar addresses. - /// The current round data. - /// The world state. - /// The list of avatar addresses, scores, and ranks. - public static List<(Address avatarAddr, int score, int rank)> AvatarAddrAndScoresWithRank(List
avatarAddrList, ArenaSheet.RoundData currentRoundData, IWorldState worldState) - { - var avatarAndScoreAddrList = avatarAddrList - .Select(avatarAddr => ( - avatarAddr, - ArenaScore.DeriveAddress( - avatarAddr, - currentRoundData.ChampionshipId, - currentRoundData.Round))) - .ToArray(); - // NOTE: If addresses is too large, and split and get separately. - var scores = worldState.GetLegacyStates( - avatarAndScoreAddrList.Select(tuple => tuple.Item2).ToList()); - var avatarAddrAndScores = new List<(Address avatarAddr, int score)>(); - for (int i = 0; i < avatarAddrList.Count; i++) - { - var tuple = avatarAndScoreAddrList[i]; - var score = scores[i] is List scoreList ? (int)(Integer)scoreList[1] : ArenaScore.ArenaScoreDefault; - avatarAddrAndScores.Add((tuple.avatarAddr, score)); - } - - List<(Address avatarAddr, int score, int rank)> orderedTuples = avatarAddrAndScores - .OrderByDescending(tuple => tuple.score) - .ThenBy(tuple => tuple.avatarAddr) - .Select(tuple => (tuple.avatarAddr, tuple.score, 0)) - .ToList(); - int? currentScore = null; - var currentRank = 1; - var avatarAddrAndScoresWithRank = new List<(Address avatarAddr, int score, int rank)>(); - var trunk = new List<(Address avatarAddr, int score, int rank)>(); - for (var i = 0; i < orderedTuples.Count; i++) - { - var tuple = orderedTuples[i]; - if (!currentScore.HasValue) - { - currentScore = tuple.score; - trunk.Add(tuple); - continue; - } - - if (currentScore.Value == tuple.score) - { - trunk.Add(tuple); - currentRank++; - if (i < orderedTuples.Count - 1) - { - continue; - } - - foreach (var tupleInTrunk in trunk) - { - avatarAddrAndScoresWithRank.Add(( - tupleInTrunk.avatarAddr, - tupleInTrunk.score, - currentRank)); - } - - trunk.Clear(); - - continue; - } - - foreach (var tupleInTrunk in trunk) - { - avatarAddrAndScoresWithRank.Add(( - tupleInTrunk.avatarAddr, - tupleInTrunk.score, - currentRank)); - } - - trunk.Clear(); - if (i < orderedTuples.Count - 1) - { - trunk.Add(tuple); - currentScore = tuple.score; - currentRank++; - continue; - } - - avatarAddrAndScoresWithRank.Add(( - tuple.avatarAddr, - tuple.score, - currentRank + 1)); - } - - return avatarAddrAndScoresWithRank; - } - - /// - /// Retrieve a list of arena participants based on the provided world state, avatar address list, and avatar addresses with scores and ranks. - /// - /// The world state from which to retrieve the arena participants. - /// The list of avatar addresses to filter the matching participants. - /// The list of avatar addresses with their scores and ranks. - /// A list of arena participants. - public static List GetArenaParticipants(IWorldState worldState, List
avatarAddrList, List<(Address avatarAddr, int score, int rank)> avatarAddrAndScoresWithRank) - { - var runeListSheet = worldState.GetSheet(); - var costumeSheet = worldState.GetSheet(); - var characterSheet = worldState.GetSheet(); - var runeOptionSheet = worldState.GetSheet(); - var runeIds = runeListSheet.Values.Select(x => x.Id).ToList(); - var row = characterSheet[GameConfig.DefaultAvatarCharacterId]; - CollectionSheet collectionSheet = new CollectionSheet(); - var collectionStates = worldState.GetCollectionStates(avatarAddrList); - bool collectionSheetExist = true; - try - { - collectionSheet = worldState.GetSheet(); - } - catch (Exception) - { - collectionSheetExist = false; - } - - var result = avatarAddrAndScoresWithRank.Select(tuple => - { - var (avatarAddr, score, rank) = tuple; - var runeStates = worldState.GetRuneState(avatarAddr, out _); - var avatar = worldState.GetAvatarState(avatarAddr, getWorldInformation: false, getQuestList: false); - var itemSlotState = - worldState.GetLegacyState(ItemSlotState.DeriveAddress(avatarAddr, BattleType.Arena)) is - List itemSlotList - ? new ItemSlotState(itemSlotList) - : new ItemSlotState(BattleType.Arena); - - var runeSlotState = - worldState.GetLegacyState(RuneSlotState.DeriveAddress(avatarAddr, BattleType.Arena)) is - List runeSlotList - ? new RuneSlotState(runeSlotList) - : new RuneSlotState(BattleType.Arena); - - var equippedRuneStates = new List(); - foreach (var runeId in runeSlotState.GetRuneSlot().Select(slot => slot.RuneId)) - { - if (!runeId.HasValue) - { - continue; - } - - if (runeStates.TryGetRuneState(runeId.Value, out var runeState)) - { - equippedRuneStates.Add(runeState); - } - } - - var equipments = itemSlotState.Equipments - .Select(guid => - avatar.inventory.Equipments.FirstOrDefault(x => x.ItemId == guid)) - .Where(item => item != null).ToList(); - var costumes = itemSlotState.Costumes - .Select(guid => - avatar.inventory.Costumes.FirstOrDefault(x => x.ItemId == guid)) - .Where(item => item != null).ToList(); - var runeOptions = StateQuery.GetRuneOptions(equippedRuneStates, runeOptionSheet); - var collectionExist = collectionStates.ContainsKey(avatarAddr); - var collectionModifiers = new List(); - if (collectionSheetExist && collectionExist) - { - var collectionState = collectionStates[avatarAddr]; - foreach (var collectionId in collectionState.Ids) - { - collectionModifiers.AddRange(collectionSheet[collectionId].StatModifiers); - } - } - - var cp = CPHelper.TotalCP(equipments, costumes, runeOptions, avatar.level, row, costumeSheet, collectionModifiers, - RuneHelper.CalculateRuneLevelBonus(runeStates, runeListSheet, worldState.GetSheet()) - ); - var portraitId = StateQuery.GetPortraitId(equipments, costumes); - return new ArenaParticipant( - avatarAddr, - score, - rank, - avatar, - portraitId, - 0, - 0, - cp - ); - }).ToList(); - return result; - } - - /// - /// Prepares the arena participants by syncing the arena cache. - /// - public void PrepareArenaParticipants() - { - _logger.Information("[ArenaParticipantsWorker]Start Sync Arena Cache"); - var sw = new Stopwatch(); - sw.Start(); - // Copy from NineChronicles RxProps.Arena - // https://github.com/planetarium/NineChronicles/blob/80.0.1/nekoyume/Assets/_Scripts/State/RxProps.Arena.cs#L279 - var blockChain = _context.BlockChain; - if (blockChain is null) - { - _logger.Warning("[ArenaParticipantsWorker]BlockChain is null"); - throw new Exception(); - } - - var tip = blockChain.Tip; - var blockIndex = blockChain.Tip.Index; - var worldState = blockChain.GetWorldState(tip.Hash); - var currentRoundData = GetRoundData(worldState, blockIndex); - var participants = GetArenaParticipantsState(worldState, currentRoundData); - var cacheKey = $"{currentRoundData.ChampionshipId}_{currentRoundData.Round}"; - if (participants is null) - { - _cache.ArenaParticipantsCache.Set(cacheKey, new List()); - _logger.Information("[ArenaParticipantsWorker] participants({CacheKey}) is null. set empty list", cacheKey); - return; - } - - var avatarAddrList = participants.AvatarAddresses; - var avatarAddrAndScoresWithRank = AvatarAddrAndScoresWithRank(avatarAddrList, currentRoundData, worldState); - var result = GetArenaParticipants(worldState, avatarAddrList, avatarAddrAndScoresWithRank); - _cache.ArenaParticipantsCache.Set(cacheKey, result, TimeSpan.FromHours(1)); - sw.Stop(); - _logger.Information("[ArenaParticipantsWorker]Set Arena Cache[{CacheKey}]: {Elapsed}", cacheKey, sw.Elapsed); - } -} diff --git a/NineChronicles.Headless/BlockChainService.cs b/NineChronicles.Headless/BlockChainService.cs index eb4dcbc72..0c2bff96f 100644 --- a/NineChronicles.Headless/BlockChainService.cs +++ b/NineChronicles.Headless/BlockChainService.cs @@ -34,7 +34,6 @@ namespace NineChronicles.Headless { public class BlockChainService : ServiceBase, IBlockChainService { - private static readonly Codec Codec = new Codec(); private BlockChain _blockChain; private Swarm _swarm; private RpcContext _context; @@ -346,7 +345,7 @@ public UnaryResult GetBalanceByStateRootHash( public UnaryResult GetTip() { Bencodex.Types.Dictionary headerDict = _blockChain.Tip.MarshalBlock(); - byte[] headerBytes = Codec.Encode(headerDict); + byte[] headerBytes = _codec.Encode(headerDict); return new UnaryResult(headerBytes); } diff --git a/NineChronicles.Headless/GraphTypes/States/CombinationSlotStateType.cs b/NineChronicles.Headless/GraphTypes/States/CombinationSlotStateType.cs index c83945f50..f798e4b59 100644 --- a/NineChronicles.Headless/GraphTypes/States/CombinationSlotStateType.cs +++ b/NineChronicles.Headless/GraphTypes/States/CombinationSlotStateType.cs @@ -13,13 +13,13 @@ public CombinationSlotStateType() description: "Address of combination slot.", resolve: context => context.Source.address); Field>( - nameof(CombinationSlotState.UnlockBlockIndex), + "unlockBlockIndex", description: "Block index at the combination slot can be usable.", - resolve: context => context.Source.UnlockBlockIndex); + resolve: context => context.Source.WorkCompleteBlockIndex); Field>( - nameof(CombinationSlotState.StartBlockIndex), + "startBlockIndex", description: "Block index at the combination started.", - resolve: context => context.Source.StartBlockIndex); + resolve: context => context.Source.WorkStartBlockIndex); Field( nameof(CombinationSlotState.PetId), description: "Pet id used in equipment",