Skip to content

Commit

Permalink
Merge pull request #2458 from planetarium/release/130
Browse files Browse the repository at this point in the history
Deploy 130 (v200170)
  • Loading branch information
U-lis authored Apr 29, 2024
2 parents c810b88 + 4d31212 commit f5acd9c
Show file tree
Hide file tree
Showing 36 changed files with 288 additions and 806 deletions.
2 changes: 1 addition & 1 deletion Lib9c
Submodule Lib9c updated 265 files
101 changes: 86 additions & 15 deletions Libplanet.Headless/Hosting/LibplanetNodeService.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
using System.IO.Compression;
using System.Linq;
using System.Net.Http;
using System.Security.Cryptography;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using Bencodex;
Expand Down Expand Up @@ -635,32 +637,85 @@ public override void Dispose()
Log.Debug("Store disposed.");
}

private string ResolvePluginPath(string path) =>
Uri.IsWellFormedUriString(path, UriKind.Absolute)
? DownloadPlugin(path).Result
: path;
private string ResolvePluginPath(string path)
{
if (Uri.IsWellFormedUriString(path, UriKind.Absolute))
{
// Run the download on a background thread
return Task.Run(() => DownloadPlugin(path)).GetAwaiter().GetResult();
}

return path;
}

private async Task<string> DownloadPlugin(string url)
{
var path = Path.Combine(Environment.CurrentDirectory, "plugins");
var basePath = Environment.GetEnvironmentVariable("PLUGIN_PATH") ?? Environment.CurrentDirectory;
var path = Path.Combine(basePath, "plugins");
Directory.CreateDirectory(path);
var hashed = url.GetHashCode().ToString();
var hashed = CreateSha256Hash(url);
var logger = Log.ForContext("LibplanetNodeService", hashed);
using var httpClient = new HttpClient();
var downloadPath = Path.Join(path, hashed + ".zip");
var extractPath = Path.Join(path, hashed);
logger.Debug("Downloading...");
await File.WriteAllBytesAsync(
downloadPath,
await httpClient.GetByteArrayAsync(url, SwarmCancellationToken),
SwarmCancellationToken);
logger.Debug("Finished downloading.");
logger.Debug("Extracting...");
ZipFile.ExtractToDirectory(downloadPath, extractPath);
logger.Debug("Finished extracting.");
var checksumPath = Path.Join(extractPath, "checksum.txt");

logger.Debug("Download Path: {downloadPath}", downloadPath);
logger.Debug("Extract Path: {extractPath}", extractPath);

if (!File.Exists(downloadPath))
{
logger.Debug("Downloading...");
var content = await httpClient.GetByteArrayAsync(url);
await File.WriteAllBytesAsync(downloadPath, content);
logger.Debug("Finished downloading.");
}
else
{
logger.Debug("Download skipped, file already exists.");
}

if (!Directory.Exists(extractPath) || (File.Exists(checksumPath) && CalculateDirectoryChecksum(extractPath) != await File.ReadAllTextAsync(checksumPath)))
{
Directory.CreateDirectory(extractPath);
logger.Debug("Extracting...");
ZipFile.ExtractToDirectory(downloadPath, extractPath, true);
logger.Debug("Finished extracting.");

// Calculate checksum of extracted files and save it
var checksum = CalculateDirectoryChecksum(extractPath);
await File.WriteAllTextAsync(checksumPath, checksum);
}
else
{
logger.Debug("Extraction skipped, folder already exists and is verified.");
}

return Path.Combine(extractPath, "Lib9c.Plugin.dll");
}

private string CalculateDirectoryChecksum(string directoryPath)
{
using var md5 = MD5.Create();
var files = Directory.GetFiles(directoryPath, "*", SearchOption.AllDirectories)
.OrderBy(p => p).ToArray();

foreach (var file in files)
{
// hash path
var pathBytes = Encoding.UTF8.GetBytes(file.Substring(directoryPath.Length));
md5.TransformBlock(pathBytes, 0, pathBytes.Length, pathBytes, 0);

// hash contents
var contentBytes = File.ReadAllBytes(file);
md5.TransformBlock(contentBytes, 0, contentBytes.Length, contentBytes, 0);
}

// Handles empty content.
md5.TransformFinalBlock(Array.Empty<byte>(), 0, 0);
return BitConverter.ToString(md5.Hash!).Replace("-", "").ToLowerInvariant();
}

// FIXME: Request libplanet provide default implementation.
private sealed class ActionTypeLoaderContext : IActionTypeLoaderContext
{
Expand All @@ -671,5 +726,21 @@ public ActionTypeLoaderContext(long index)

public long Index { get; }
}

private static string CreateSha256Hash(string input)
{
using SHA256 sha256Hash = SHA256.Create();

// ComputeHash - returns byte array
byte[] bytes = sha256Hash.ComputeHash(Encoding.UTF8.GetBytes(input));

// Convert byte array to a string
StringBuilder builder = new StringBuilder();
foreach (var t in bytes)
{
builder.Append(t.ToString("x2"));
}
return builder.ToString();
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@

<ItemGroup>
<ProjectReference Include="..\NineChronicles.Headless.Executable\NineChronicles.Headless.Executable.csproj" />
<ProjectReference Include="..\Lib9c\.Libplanet\Libplanet.Mocks\Libplanet.Mocks.csproj" />
</ItemGroup>

</Project>
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,11 @@ public void Balance(
_console.Error.WriteLine("Scanning block #{0} {1}...", digest.Index, digest.Hash);
_console.Error.Flush();
IEnumerable<Address> addrs = digest.TxIds
.Select(txId => store.GetTransaction(new TxId(txId.ToArray())))
.Select(txId =>
{
return store.GetTransaction(new TxId(txId.ToArray())) ??
throw new InvalidOperationException($"Transaction #{txId} is not found in the store.");
})
.SelectMany(tx => tx.Actions is { } ca
? ca.Select(a => ToAction(a))
.SelectMany(a =>
Expand Down
56 changes: 41 additions & 15 deletions NineChronicles.Headless.Executable/Commands/ChainCommand.cs
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,7 @@ public void Tip(

BlockHash tipHash = store.IndexBlockHash(chainId, -1)
?? throw new CommandExitedException("The given chain seems empty.", -1);
Block tip = store.GetBlock(tipHash);
Block tip = GetBlock(store, tipHash);
_console.Out.WriteLine(CoconaUtils.SerializeHumanReadable(tip.Header));
store.Dispose();
}
Expand Down Expand Up @@ -129,7 +129,7 @@ public void Inspect(
throw new CommandExitedException($"There is no genesis block: {storePath}", -1);
}

Block genesisBlock = store.GetBlock(gHash);
Block genesisBlock = GetBlock(store, gHash);
var blockChainStates = new BlockChainStates(store, stateStore);
var actionEvaluator = new ActionEvaluator(
_ => blockPolicy.BlockAction,
Expand Down Expand Up @@ -163,8 +163,8 @@ public void Inspect(
foreach (var item in
store.IterateIndexes(chain.Id, offset + 1 ?? 1, limit).Select((value, i) => new { i, value }))
{
var block = store.GetBlock(item.value);
var previousBlock = store.GetBlock(
var block = GetBlock(store, item.value);
var previousBlock = GetBlock(store,
block.PreviousHash ?? block.Hash
);

Expand Down Expand Up @@ -270,7 +270,7 @@ public void Truncate(
-1);
}

var tip = store.GetBlock(tipHash);
var tip = GetBlock(store, tipHash);
var snapshotTipIndex = Math.Max(tipIndex - (blocksBefore + 1), 0);
BlockHash snapshotTipHash;

Expand All @@ -286,7 +286,7 @@ public void Truncate(
}

snapshotTipHash = hash;
} while (!stateStore.GetStateRoot(store.GetBlock(snapshotTipHash).StateRootHash).Recorded);
} while (!stateStore.GetStateRoot(GetBlock(store, snapshotTipHash).StateRootHash).Recorded);

var forkedId = Guid.NewGuid();

Expand Down Expand Up @@ -484,11 +484,11 @@ public void Snapshot(
);
var originalChain = new BlockChain(blockPolicy, stagePolicy, store, stateStore,
store.GetBlock(genesisHash), blockChainStates, actionEvaluator);
var tip = store.GetBlock(tipHash);
var tip = GetBlock(store, tipHash);

var potentialSnapshotTipIndex = tipIndex - blockBefore;
var potentialSnapshotTipHash = (BlockHash)store.IndexBlockHash(chainId, potentialSnapshotTipIndex)!;
var snapshotTip = store.GetBlock(potentialSnapshotTipHash);
var snapshotTip = GetBlock(store, potentialSnapshotTipHash);

_console.Out.WriteLine(
"Original Store Tip: #{0}\n1. LastCommit: {1}\n2. BlockCommit in Chain: {2}\n3. BlockCommit in Store: {3}",
Expand Down Expand Up @@ -520,24 +520,38 @@ public void Snapshot(
"There is no block commit associated with the potential snapshot tip: #{0}. Snapshot will automatically truncate 1 more block from the original chain tip.",
potentialSnapshotTipIndex);
blockBefore += 1;
potentialSnapshotTipBlockCommit = store
.GetBlock((BlockHash)store.IndexBlockHash(chainId, tip.Index - blockBefore + 1)!).LastCommit;
potentialSnapshotTipBlockCommit =
GetBlock(store, (BlockHash)store.IndexBlockHash(chainId, tip.Index - blockBefore + 1)!).LastCommit;
if (potentialSnapshotTipBlockCommit == null)
{
throw new CommandExitedException(
$"The block commit of the potential snapshot tip: #{potentialSnapshotTipIndex} doesn't exist.",
-1);
}

store.PutBlockCommit(tipBlockCommit);
store.PutChainBlockCommit(chainId, tipBlockCommit);
store.PutBlockCommit(potentialSnapshotTipBlockCommit);
store.PutChainBlockCommit(chainId, potentialSnapshotTipBlockCommit);
}

var blockCommitBlock = store.GetBlock(tipHash);
var blockCommitBlock = GetBlock(store, tipHash);

// Add last block commits to store from tip until --block-before + 5 or tip if too short for buffer
var blockCommitRange = blockBefore + 5 < tip.Index ? blockBefore + 5 : tip.Index - 1;
for (var i = 0; i < blockCommitRange; i++)
{
_console.Out.WriteLine("Adding block #{0}'s block commit to the store", blockCommitBlock.Index - 1);
if (blockCommitBlock.LastCommit == null)
{
throw new CommandExitedException(
$"The block commit of the block: #{blockCommitBlock.Index} doesn't exist.",
-1);
}

store.PutBlockCommit(blockCommitBlock.LastCommit);
store.PutChainBlockCommit(chainId, blockCommitBlock.LastCommit);
blockCommitBlock = store.GetBlock((BlockHash)blockCommitBlock.PreviousHash!);
blockCommitBlock = GetBlock(store, (BlockHash)blockCommitBlock.PreviousHash!);
}

var snapshotTipIndex = Math.Max(tipIndex - (blockBefore + 1), 0);
Expand All @@ -555,7 +569,7 @@ public void Snapshot(
}

snapshotTipHash = hash;
} while (!stateStore.GetStateRoot(store.GetBlock(snapshotTipHash).StateRootHash).Recorded);
} while (!stateStore.GetStateRoot(GetBlock(store, snapshotTipHash).StateRootHash).Recorded);

var forkedId = Guid.NewGuid();
Fork(chainId, forkedId, snapshotTipHash, tip, store);
Expand Down Expand Up @@ -745,6 +759,18 @@ public void Snapshot(
}
}

private static Block GetBlock(IStore store, BlockHash blockHash)
{
return store.GetBlock(blockHash) ??
throw new CommandExitedException($"The block of {blockHash} doesn't exist.", -1);
}

private static BlockCommit GetBlockCommit(IStore store, BlockHash blockHash)
{
return store.GetBlockCommit(blockHash) ??
throw new CommandExitedException($"The block commit of {blockHash} doesn't exist.", -1);
}

private string GetPartitionBaseFileName(
int currentMetadataBlockEpoch,
int currentMetadataTxEpoch,
Expand Down Expand Up @@ -1025,15 +1051,15 @@ private void Fork(
store.ForkBlockIndexes(src, dest, branchPointHash);
if (store.GetBlockCommit(branchPointHash) is { })
{
store.PutChainBlockCommit(dest, store.GetBlockCommit(branchPointHash));
store.PutChainBlockCommit(dest, GetBlockCommit(store, branchPointHash));
}
store.ForkTxNonces(src, dest);

for (
Block block = tip;
block.PreviousHash is { } hash
&& !block.Hash.Equals(branchPointHash);
block = store.GetBlock(hash))
block = GetBlock(store, hash))
{
IEnumerable<(Address, int)> signers = block
.Transactions
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,9 @@ public ActionContext(

public bool BlockAction => TxId is null;

// NOTE: Replay does not support block actions.
public IReadOnlyList<ITransaction> Txs => ImmutableList<ITransaction>.Empty;

public void UseGas(long gas)
{
}
Expand Down
3 changes: 2 additions & 1 deletion NineChronicles.Headless.Executable/Commands/StateCommand.cs
Original file line number Diff line number Diff line change
Expand Up @@ -121,7 +121,8 @@ IStateStore stateStore
}

Block block =
store.GetBlock(blockHash);
store.GetBlock(blockHash) ??
throw new CommandExitedException($"The block of {blockHash} doesn't exist.", -1);
var preEvalBlock = new PreEvaluationBlock(
block,
block.Transactions
Expand Down
7 changes: 6 additions & 1 deletion NineChronicles.Headless.Executable/Store/StoreExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,12 @@ public static Block GetGenesisBlock(this IStore store)
}

BlockHash genesisBlockHash = store.IterateIndexes(chainId.Value).First();
Block genesisBlock = store.GetBlock(genesisBlockHash);
Block? genesisBlock = store.GetBlock(genesisBlockHash);
if (genesisBlock == null)
{
throw new InvalidOperationException("The store doesn't have genesis block.");
}

return genesisBlock;
}
}
Expand Down
10 changes: 4 additions & 6 deletions NineChronicles.Headless.Tests/ArenaParticipantsWorkerTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
using Lib9c.Tests;
using Libplanet.Action.State;
using Libplanet.Crypto;
using Libplanet.Mocks;
using Nekoyume;
using Nekoyume.Action;
using Nekoyume.Model.Arena;
Expand All @@ -12,7 +13,6 @@
using Nekoyume.Model.State;
using Nekoyume.Module;
using Nekoyume.TableData;
using NineChronicles.Headless.Tests.Common;
using Xunit;
using Random = Libplanet.Extensions.ActionEvaluatorCommonComponents.Random;

Expand All @@ -25,7 +25,7 @@ public class ArenaParticipantsWorkerTest

public ArenaParticipantsWorkerTest()
{
_world = new MockWorld(new MockWorldState());
_world = new World(MockWorldState.CreateModern());
_sheets = TableSheetsImporter.ImportSheets();
}

Expand Down Expand Up @@ -115,7 +115,6 @@ public void GetArenaParticipants()
agentAddress,
0,
tableSheets.GetAvatarSheets(),
new GameConfigState(),
new Address(),
"avatar_state"
);
Expand All @@ -125,7 +124,6 @@ public void GetArenaParticipants()
agentAddress,
0,
tableSheets.GetAvatarSheets(),
new GameConfigState(),
new Address(),
"avatar_state2"
);
Expand Down Expand Up @@ -154,7 +152,7 @@ public void GetArenaParticipants()
{
runeSlotInfo,
}, runeListSheet);
var runeState = new RuneState(runeId);
var runeStates = new AllRuneState(runeId);

// collection
var collectionSheet = tableSheets.CollectionSheet;
Expand All @@ -175,7 +173,7 @@ public void GetArenaParticipants()
.SetAvatarState(avatarAddress, avatarState, true, true, true, true)
.SetAvatarState(avatar2Address, avatarState2, true, true, true, true)
.SetLegacyState(itemSlotAddress, itemSlotState.Serialize())
.SetLegacyState(RuneState.DeriveAddress(avatarAddress, runeId), runeState.Serialize())
.SetRuneState(avatarAddress, runeStates)
.SetLegacyState(runeSlotAddress, runeSlotState.Serialize())
.SetCollectionState(avatar2Address, collectionState)
.SetLegacyState(participantsAddr, participants.Serialize())
Expand Down
Loading

0 comments on commit f5acd9c

Please sign in to comment.