com.hedera.node.app.service.file.impl.handlers.FileDeleteHandler Maven / Gradle / Ivy
Show all versions of app-service-file-impl Show documentation
/*
* Copyright (C) 2022-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.node.app.service.file.impl.handlers;
import static com.hedera.hapi.node.base.ResponseCodeEnum.INVALID_FILE_ID;
import static com.hedera.hapi.node.base.ResponseCodeEnum.UNAUTHORIZED;
import static com.hedera.node.app.service.file.impl.utils.FileServiceUtils.preValidate;
import static com.hedera.node.app.service.file.impl.utils.FileServiceUtils.validateAndAddRequiredKeysForDelete;
import static com.hedera.node.app.service.file.impl.utils.FileServiceUtils.verifyNotSystemFile;
import static com.hedera.node.app.spi.workflows.HandleException.validateFalse;
import static java.util.Objects.requireNonNull;
import com.hedera.hapi.node.base.HederaFunctionality;
import com.hedera.hapi.node.base.SubType;
import com.hedera.hapi.node.file.FileDeleteTransactionBody;
import com.hedera.hapi.node.state.file.File;
import com.hedera.hapi.node.transaction.TransactionBody;
import com.hedera.node.app.hapi.utils.CommonPbjConverters;
import com.hedera.node.app.hapi.utils.fee.FileFeeBuilder;
import com.hedera.node.app.service.file.ReadableFileStore;
import com.hedera.node.app.service.file.impl.WritableFileStore;
import com.hedera.node.app.spi.fees.FeeContext;
import com.hedera.node.app.spi.fees.Fees;
import com.hedera.node.app.spi.workflows.HandleContext;
import com.hedera.node.app.spi.workflows.HandleException;
import com.hedera.node.app.spi.workflows.PreCheckException;
import com.hedera.node.app.spi.workflows.PreHandleContext;
import com.hedera.node.app.spi.workflows.TransactionHandler;
import com.hedera.node.config.data.LedgerConfig;
import com.hedera.pbj.runtime.io.buffer.Bytes;
import edu.umd.cs.findbugs.annotations.NonNull;
import javax.inject.Inject;
import javax.inject.Singleton;
/**
* This class contains all workflow-related functionality regarding {@link HederaFunctionality#FILE_DELETE}.
*/
@Singleton
public class FileDeleteHandler implements TransactionHandler {
private final FileFeeBuilder usageEstimator;
/**
* Constructs a {@link FileDeleteHandler} with the given {@link FileFeeBuilder}.
*
* @param usageEstimator the file fee builder to be used for fee calculation
*/
@Inject
public FileDeleteHandler(final FileFeeBuilder usageEstimator) {
this.usageEstimator = usageEstimator;
}
/**
* Performs checks independent of state or context.
*
* @param txn the transaction to check
*/
@Override
public void pureChecks(@NonNull final TransactionBody txn) throws PreCheckException {
final FileDeleteTransactionBody transactionBody = txn.fileDeleteOrThrow();
if (transactionBody.fileID() == null) {
throw new PreCheckException(INVALID_FILE_ID);
}
}
/**
* This method is called during the pre-handle workflow.
*
* Determines signatures needed for deleting a file
*
* @param context the {@link PreHandleContext} which collects all information that will be passed to
* {@code handle()}
* @throws PreCheckException if any issue happens on the pre handle level
*/
@Override
public void preHandle(@NonNull final PreHandleContext context) throws PreCheckException {
requireNonNull(context);
final var transactionBody = context.body().fileDeleteOrThrow();
final var fileStore = context.createStore(ReadableFileStore.class);
final var transactionFileId = requireNonNull(transactionBody.fileID());
preValidate(transactionFileId, fileStore, context);
var file = fileStore.getFileLeaf(transactionFileId);
validateAndAddRequiredKeysForDelete(file, context);
}
@Override
public void handle(@NonNull final HandleContext handleContext) throws HandleException {
requireNonNull(handleContext);
final var fileDeleteTransactionBody = handleContext.body().fileDeleteOrThrow();
if (!fileDeleteTransactionBody.hasFileID()) {
throw new HandleException(INVALID_FILE_ID);
}
var fileId = fileDeleteTransactionBody.fileIDOrThrow();
final var ledgerConfig = handleContext.configuration().getConfigData(LedgerConfig.class);
final var fileStore = handleContext.storeFactory().writableStore(WritableFileStore.class);
final File file = verifyNotSystemFile(ledgerConfig, fileStore, fileId);
// First validate this file is mutable; and the pending mutations are allowed
validateFalse(file.keys() == null, UNAUTHORIZED);
/* Copy part of the fields from existing, delete the file content and set the deleted flag */
final var fileBuilder = new File.Builder()
.fileId(file.fileId())
.expirationSecond(file.expirationSecond())
.keys(file.keys())
.contents(Bytes.EMPTY)
.memo(file.memo())
.deleted(true);
/* --- Put the modified file. It will be in underlying state's modifications map.
It will not be committed to state until commit is called on the state.--- */
fileStore.put(fileBuilder.build());
}
@NonNull
@Override
public Fees calculateFees(@NonNull FeeContext feeContext) {
final var txnBody = feeContext.body();
return feeContext
.feeCalculatorFactory()
.feeCalculator(SubType.DEFAULT)
.legacyCalculate(
svo -> usageEstimator.getFileDeleteTxFeeMatrices(CommonPbjConverters.fromPbj(txnBody), svo));
}
}