Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Simulator blockstream manager #169

Merged
merged 5 commits into from
Sep 10, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -56,3 +56,5 @@ server/docker/.env
# manual test files
server/src/test/resources/test_output/

# simulator files, this files are un-tarred before build
/simulator/src/main/resources/block-0.0.3/
AlfredoG87 marked this conversation as resolved.
Show resolved Hide resolved
20 changes: 20 additions & 0 deletions simulator/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -38,3 +38,23 @@ testModuleInfo {
requires("org.mockito.junit.jupiter")
requiresStatic("com.github.spotbugs.annotations")
}

tasks.register<Copy>("untarTestBlockStream") {
description = "Untar the test block stream data"
group = "build"

val targetDir = file("src/main/resources")

from(tarTree(resources.gzip(file("src/main/resources/block-0.0.3.tar.gz"))))
into(targetDir)

// Mark task as not up-to-date if the directory is empty
outputs.upToDateWhen { targetDir.listFiles()?.isNotEmpty() ?: false }

// Adding a simple logging to verify
doLast { println("Untar task completed. Files should be in: ${targetDir.absolutePath}") }
}

tasks.named("processResources") { dependsOn(tasks.named("untarTestBlockStream")) }

tasks.named("sourcesJar") { dependsOn(tasks.named("untarTestBlockStream")) }
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ public class BlockStreamSimulatorApp {

Configuration configuration;
BlockStreamManager blockStreamManager;

boolean isRunning = false;

/**
Expand All @@ -52,6 +53,17 @@ public void start() {
// use PublishStreamGrpcClient to stream it to the block-node.
isRunning = true;
LOGGER.log(System.Logger.Level.INFO, "Block Stream Simulator has started");

// while

// get block item
// send block item

// verify if ack is needed
// wait for ack async...

// verify exit condition

}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,12 +19,45 @@
import com.hedera.block.simulator.config.types.GenerationMode;
import com.swirlds.config.api.ConfigData;
import com.swirlds.config.api.ConfigProperty;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;

/**
* The BlockStreamConfig class defines the configuration data for the block stream.
*
* @param generationMode the mode of generation for the block stream
* @param folderRootPath the root path of the folder containing the block stream
*/
@ConfigData("blockStream")
public record BlockStreamConfig(
@ConfigProperty(defaultValue = "DIR") GenerationMode generationMode) {}
@ConfigProperty(defaultValue = "DIR") GenerationMode generationMode,
@ConfigProperty(defaultValue = "") String folderRootPath) {

/**
* Constructor to set the default root path if not provided, it will be set to the data
* directory in the current working directory
*/
public BlockStreamConfig {
// verify rootPath prop
Path path = Path.of(folderRootPath);

// if rootPath is empty, set it to the default data directory
if (folderRootPath.isEmpty()) {
path =
Paths.get(folderRootPath)
.toAbsolutePath()
.resolve("src/main/resources/block-0.0.3");
}
// Check if absolute
if (!path.isAbsolute()) {
throw new IllegalArgumentException(folderRootPath + " Root path must be absolute");
}
// Check if the folder exists
if (Files.notExists(path) && generationMode == GenerationMode.DIR) {
throw new IllegalArgumentException("Folder does not exist: " + path);
}

folderRootPath = path.toString();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
/*
* Copyright (C) 2024 Hedera Hashgraph, LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package com.hedera.block.simulator.generator;

import static java.lang.System.Logger.Level.DEBUG;
import static java.lang.System.Logger.Level.ERROR;
import static java.lang.System.Logger.Level.INFO;

import com.hedera.block.simulator.config.data.BlockStreamConfig;
import com.hedera.block.simulator.config.types.GenerationMode;
import com.hedera.hapi.block.stream.Block;
import com.hedera.hapi.block.stream.BlockItem;
import com.hedera.pbj.runtime.ParseException;
import com.hedera.pbj.runtime.io.buffer.Bytes;
import edu.umd.cs.findbugs.annotations.NonNull;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.stream.Stream;
import javax.inject.Inject;

/** The block as file block stream manager. */
public class BlockAsFileBlockStreamManager implements BlockStreamManager {

private final System.Logger LOGGER = System.getLogger(getClass().getName());

final String rootFolder;

final List<Block> blocks = new ArrayList<>();

int currentBlockIndex = 0;
int currentBlockItemIndex = 0;
int lastGivenBlockNumber = 0;

/**
* Constructor for the block as file block stream manager.
*
* @param blockStreamConfig the block stream config
*/
@Inject
public BlockAsFileBlockStreamManager(@NonNull BlockStreamConfig blockStreamConfig) {
this.rootFolder = blockStreamConfig.folderRootPath();
try {
this.loadBlocks();
} catch (IOException | ParseException | IllegalArgumentException e) {
LOGGER.log(ERROR, "Error loading blocks", e);
throw new RuntimeException(e);
}

LOGGER.log(INFO, "Loaded " + blocks.size() + " blocks into memory");
}

@Override
public GenerationMode getGenerationMode() {
return GenerationMode.DIR;
}

@Override
public BlockItem getNextBlockItem() {
BlockItem nextBlockItem = blocks.get(currentBlockIndex).items().get(currentBlockItemIndex);
currentBlockItemIndex++;
if (currentBlockItemIndex >= blocks.get(currentBlockIndex).items().size()) {
currentBlockItemIndex = 0;
currentBlockIndex++;
}
return nextBlockItem;
}

@Override
public Block getNextBlock() {
Block nextBlock = blocks.get(currentBlockIndex);
currentBlockIndex++;
lastGivenBlockNumber++;
if (currentBlockIndex >= blocks.size()) {
currentBlockIndex = 0;
}
return nextBlock;
}

private void loadBlocks() throws IOException, ParseException {

Path rootPath = Path.of(rootFolder);

try (Stream<Path> blockFiles = Files.list(rootPath)) {

List<Path> sortedBlockFiles =
blockFiles.sorted(Comparator.comparing(Path::getFileName)).toList();
mattp-swirldslabs marked this conversation as resolved.
Show resolved Hide resolved

for (Path blockPath : sortedBlockFiles) {

byte[] blockBytes;
if (blockPath.toString().endsWith(".gz")) {
blockBytes = Utils.readGzFile(blockPath);
} else if (blockPath.toString().endsWith(".blk")) {
blockBytes = Files.readAllBytes(blockPath);
} else {
throw new IllegalArgumentException("Invalid file format: " + blockPath);
}

Block block = Block.PROTOBUF.parse(Bytes.wrap(blockBytes));
blocks.add(block);
LOGGER.log(DEBUG, "Loaded block: " + blockPath);
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -16,15 +16,31 @@

package com.hedera.block.simulator.generator;

import com.hedera.block.simulator.config.types.GenerationMode;
import com.hedera.hapi.block.stream.Block;
import com.hedera.hapi.block.stream.BlockItem;

/** this interface defines the contract for managing the block stream. */
/** The block stream manager interface. */
public interface BlockStreamManager {

/**
* Gets the next block in the block stream.
* Get the generation mode.
*
* @return the next block in the block stream
* @return the generation mode
*/
GenerationMode getGenerationMode();

/**
* Get the next block item.
*
* @return the next block item
*/
BlockItem getNextBlockItem();

/**
* Get the next block.
*
* @return the next block
*/
Block getNextBlock();
mattp-swirldslabs marked this conversation as resolved.
Show resolved Hide resolved
}
Original file line number Diff line number Diff line change
Expand Up @@ -27,10 +27,11 @@ public interface GeneratorInjectionModule {
/**
* Provides the block stream manager.
*
* @param blockStreamManager the block stream manager to be used
* @param blockAsFileBlockStreamManager the block as file block stream manager
* @return the block stream manager
*/
@Singleton
@Binds
BlockStreamManager bindBlockStreamManager(MockBlockStreamManagerImpl blockStreamManager);
BlockStreamManager provideBlockStreamManager(
BlockAsFileBlockStreamManager blockAsFileBlockStreamManager);
}

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
/*
* Copyright (C) 2024 Hedera Hashgraph, LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package com.hedera.block.simulator.generator;

import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.zip.GZIPInputStream;

/** Utility class for the simulator. */
public final class Utils {

private Utils() {}

/**
* Read a GZIP file and return the content as a byte array.
*
* @param filePath Path to the GZIP file
* @return byte array of the content of the GZIP file
* @throws IOException if an I/O error occurs
*/
public static byte[] readGzFile(Path filePath) throws IOException {
try (InputStream fileInputStream = Files.newInputStream(filePath);
GZIPInputStream gzipInputStream = new GZIPInputStream(fileInputStream)) {
return gzipInputStream.readAllBytes();
}
}
}
2 changes: 1 addition & 1 deletion simulator/src/main/java/module-info.java
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
requires static com.github.spotbugs.annotations;
requires static com.google.auto.service;
requires com.hedera.block.stream;
// requires com.hedera.pbj.runtime; // leaving it here since it will be needed soon.
requires com.hedera.pbj.runtime;
requires com.swirlds.config.api;
requires com.swirlds.config.extensions;
requires dagger;
Expand Down
2 changes: 1 addition & 1 deletion simulator/src/main/resources/app.properties
Original file line number Diff line number Diff line change
@@ -1 +1 @@
blockStream.folderRootPath=/Users/user/Projects/hedera-block-node/simulator/src/main/resources/block-0.0.3/

AlfredoG87 marked this conversation as resolved.
Show resolved Hide resolved
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file added simulator/src/main/resources/block-0.0.3.tar.gz
Binary file not shown.
Loading
Loading