All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.iceberg.spark.actions.BaseExpireSnapshotsSparkAction Maven / Gradle / Ivy

There is a newer version: 0.13.2
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.iceberg.spark.actions;

import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.atomic.AtomicLong;
import java.util.function.Consumer;
import org.apache.iceberg.HasTableOperations;
import org.apache.iceberg.Table;
import org.apache.iceberg.TableMetadata;
import org.apache.iceberg.TableOperations;
import org.apache.iceberg.actions.BaseExpireSnapshotsActionResult;
import org.apache.iceberg.actions.ExpireSnapshots;
import org.apache.iceberg.exceptions.NotFoundException;
import org.apache.iceberg.exceptions.ValidationException;
import org.apache.iceberg.relocated.com.google.common.base.Joiner;
import org.apache.iceberg.relocated.com.google.common.base.Preconditions;
import org.apache.iceberg.relocated.com.google.common.collect.Lists;
import org.apache.iceberg.relocated.com.google.common.collect.Sets;
import org.apache.iceberg.spark.JobGroupInfo;
import org.apache.iceberg.util.PropertyUtil;
import org.apache.iceberg.util.Tasks;
import org.apache.spark.sql.Column;
import org.apache.spark.sql.Dataset;
import org.apache.spark.sql.Row;
import org.apache.spark.sql.SparkSession;
import org.apache.spark.sql.functions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import static org.apache.iceberg.TableProperties.GC_ENABLED;
import static org.apache.iceberg.TableProperties.GC_ENABLED_DEFAULT;

/**
 * An action that performs the same operation as {@link org.apache.iceberg.ExpireSnapshots} but uses Spark
 * to determine the delta in files between the pre and post-expiration table metadata. All of the same
 * restrictions of {@link org.apache.iceberg.ExpireSnapshots} also apply to this action.
 * 

* This action first leverages {@link org.apache.iceberg.ExpireSnapshots} to expire snapshots and then * uses metadata tables to find files that can be safely deleted. This is done by anti-joining two Datasets * that contain all manifest and data files before and after the expiration. The snapshot expiration * will be fully committed before any deletes are issued. *

* This operation performs a shuffle so the parallelism can be controlled through 'spark.sql.shuffle.partitions'. *

* Deletes are still performed locally after retrieving the results from the Spark executors. */ @SuppressWarnings("UnnecessaryAnonymousClass") public class BaseExpireSnapshotsSparkAction extends BaseSparkAction implements ExpireSnapshots { private static final Logger LOG = LoggerFactory.getLogger(BaseExpireSnapshotsSparkAction.class); private static final String DATA_FILE = "Data File"; private static final String MANIFEST = "Manifest"; private static final String MANIFEST_LIST = "Manifest List"; private static final String STREAM_RESULTS = "stream-results"; // Creates an executor service that runs each task in the thread that invokes execute/submit. private static final ExecutorService DEFAULT_DELETE_EXECUTOR_SERVICE = null; private final Table table; private final TableOperations ops; private final Consumer defaultDelete = new Consumer() { @Override public void accept(String file) { ops.io().deleteFile(file); } }; private final Set expiredSnapshotIds = Sets.newHashSet(); private Long expireOlderThanValue = null; private Integer retainLastValue = null; private Consumer deleteFunc = defaultDelete; private ExecutorService deleteExecutorService = DEFAULT_DELETE_EXECUTOR_SERVICE; private Dataset expiredFiles = null; public BaseExpireSnapshotsSparkAction(SparkSession spark, Table table) { super(spark); this.table = table; this.ops = ((HasTableOperations) table).operations(); ValidationException.check( PropertyUtil.propertyAsBoolean(table.properties(), GC_ENABLED, GC_ENABLED_DEFAULT), "Cannot expire snapshots: GC is disabled (deleting files may corrupt other tables)"); } @Override protected ExpireSnapshots self() { return this; } @Override public BaseExpireSnapshotsSparkAction executeDeleteWith(ExecutorService executorService) { this.deleteExecutorService = executorService; return this; } @Override public BaseExpireSnapshotsSparkAction expireSnapshotId(long snapshotId) { expiredSnapshotIds.add(snapshotId); return this; } @Override public BaseExpireSnapshotsSparkAction expireOlderThan(long timestampMillis) { this.expireOlderThanValue = timestampMillis; return this; } @Override public BaseExpireSnapshotsSparkAction retainLast(int numSnapshots) { Preconditions.checkArgument(1 <= numSnapshots, "Number of snapshots to retain must be at least 1, cannot be: %s", numSnapshots); this.retainLastValue = numSnapshots; return this; } @Override public BaseExpireSnapshotsSparkAction deleteWith(Consumer newDeleteFunc) { this.deleteFunc = newDeleteFunc; return this; } /** * Expires snapshots and commits the changes to the table, returning a Dataset of files to delete. *

* This does not delete data files. To delete data files, run {@link #execute()}. *

* This may be called before or after {@link #execute()} is called to return the expired file list. * * @return a Dataset of files that are no longer referenced by the table */ public Dataset expire() { if (expiredFiles == null) { // fetch metadata before expiration Dataset originalFiles = buildValidFileDF(ops.current()); // perform expiration org.apache.iceberg.ExpireSnapshots expireSnapshots = table.expireSnapshots().cleanExpiredFiles(false); for (long id : expiredSnapshotIds) { expireSnapshots = expireSnapshots.expireSnapshotId(id); } if (expireOlderThanValue != null) { expireSnapshots = expireSnapshots.expireOlderThan(expireOlderThanValue); } if (retainLastValue != null) { expireSnapshots = expireSnapshots.retainLast(retainLastValue); } expireSnapshots.commit(); // fetch metadata after expiration Dataset validFiles = buildValidFileDF(ops.refresh()); // determine expired files this.expiredFiles = originalFiles.except(validFiles); } return expiredFiles; } @Override public ExpireSnapshots.Result execute() { JobGroupInfo info = newJobGroupInfo("EXPIRE-SNAPSHOTS", jobDesc()); return withJobGroupInfo(info, this::doExecute); } private String jobDesc() { List options = Lists.newArrayList(); if (expireOlderThanValue != null) { options.add("older_than=" + expireOlderThanValue); } if (retainLastValue != null) { options.add("retain_last=" + retainLastValue); } if (!expiredSnapshotIds.isEmpty()) { Long first = expiredSnapshotIds.stream().findFirst().get(); if (expiredSnapshotIds.size() > 1) { options.add(String.format("snapshot_ids: %s (%s more...)", first, expiredSnapshotIds.size() - 1)); } else { options.add(String.format("snapshot_id: %s", first)); } } return String.format("Expiring snapshots (%s) in %s", Joiner.on(',').join(options), table.name()); } private ExpireSnapshots.Result doExecute() { boolean streamResults = PropertyUtil.propertyAsBoolean(options(), STREAM_RESULTS, false); if (streamResults) { return deleteFiles(expire().toLocalIterator()); } else { return deleteFiles(expire().collectAsList().iterator()); } } private Dataset appendTypeString(Dataset ds, String type) { return ds.select(new Column("file_path"), functions.lit(type).as("file_type")); } private Dataset buildValidFileDF(TableMetadata metadata) { Table staticTable = newStaticTable(metadata, this.table.io()); return appendTypeString(buildValidDataFileDF(staticTable), DATA_FILE) .union(appendTypeString(buildManifestFileDF(staticTable), MANIFEST)) .union(appendTypeString(buildManifestListDF(staticTable), MANIFEST_LIST)); } /** * Deletes files passed to it based on their type. * * @param expired an Iterator of Spark Rows of the structure (path: String, type: String) * @return Statistics on which files were deleted */ private BaseExpireSnapshotsActionResult deleteFiles(Iterator expired) { AtomicLong dataFileCount = new AtomicLong(0L); AtomicLong manifestCount = new AtomicLong(0L); AtomicLong manifestListCount = new AtomicLong(0L); Tasks.foreach(expired) .retry(3).stopRetryOn(NotFoundException.class).suppressFailureWhenFinished() .executeWith(deleteExecutorService) .onFailure((fileInfo, exc) -> { String file = fileInfo.getString(0); String type = fileInfo.getString(1); LOG.warn("Delete failed for {}: {}", type, file, exc); }) .run(fileInfo -> { String file = fileInfo.getString(0); String type = fileInfo.getString(1); deleteFunc.accept(file); switch (type) { case DATA_FILE: dataFileCount.incrementAndGet(); LOG.trace("Deleted Data File: {}", file); break; case MANIFEST: manifestCount.incrementAndGet(); LOG.debug("Deleted Manifest: {}", file); break; case MANIFEST_LIST: manifestListCount.incrementAndGet(); LOG.debug("Deleted Manifest List: {}", file); break; } }); LOG.info("Deleted {} total files", dataFileCount.get() + manifestCount.get() + manifestListCount.get()); return new BaseExpireSnapshotsActionResult(dataFileCount.get(), manifestCount.get(), manifestListCount.get()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy