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

org.apache.iceberg.dell.ecs.EcsAppendOutputStream Maven / Gradle / Ivy

There is a newer version: 1.0.0.8
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.dell.ecs;

import com.emc.object.s3.S3Client;
import com.emc.object.s3.request.PutObjectRequest;
import java.io.ByteArrayInputStream;
import java.nio.ByteBuffer;
import org.apache.iceberg.io.FileIOMetricsContext;
import org.apache.iceberg.io.PositionOutputStream;
import org.apache.iceberg.metrics.MetricsContext;
import org.apache.iceberg.metrics.MetricsContext.Counter;
import org.apache.iceberg.metrics.MetricsContext.Unit;

/** Use ECS append API to write data. */
class EcsAppendOutputStream extends PositionOutputStream {

  private final S3Client client;

  private final EcsURI uri;

  /**
   * Local bytes cache that avoid too many requests
   *
   * 

Use {@link ByteBuffer} to maintain offset. */ private final ByteBuffer localCache; /** A marker for data file to put first part instead of append first part. */ private boolean firstPart = true; /** Pos for {@link PositionOutputStream} */ private long pos; private final Counter writeBytes; private final Counter writeOperations; private EcsAppendOutputStream( S3Client client, EcsURI uri, byte[] localCache, MetricsContext metrics) { this.client = client; this.uri = uri; this.localCache = ByteBuffer.wrap(localCache); this.writeBytes = metrics.counter(FileIOMetricsContext.WRITE_BYTES, Long.class, Unit.BYTES); this.writeOperations = metrics.counter(FileIOMetricsContext.WRITE_OPERATIONS, Integer.class, Unit.COUNT); } /** Use built-in 1 KiB byte buffer */ static EcsAppendOutputStream create(S3Client client, EcsURI uri, MetricsContext metrics) { return createWithBufferSize(client, uri, 1024, metrics); } /** Create {@link PositionOutputStream} with specific buffer size. */ static EcsAppendOutputStream createWithBufferSize( S3Client client, EcsURI uri, int size, MetricsContext metrics) { return new EcsAppendOutputStream(client, uri, new byte[size], metrics); } /** Write a byte. If buffer is full, upload the buffer. */ @Override public void write(int b) { if (!checkBuffer(1)) { flush(); } localCache.put((byte) b); pos += 1; writeBytes.increment(); writeOperations.increment(); } /** * Write a byte. If buffer is full, upload the buffer. If buffer size < input bytes, upload * input bytes. */ @Override public void write(byte[] b, int off, int len) { if (!checkBuffer(len)) { flush(); } if (checkBuffer(len)) { localCache.put(b, off, len); } else { // if content > cache, directly flush itself. flushBuffer(b, off, len); } pos += len; writeBytes.increment((long) len); writeOperations.increment(); } private boolean checkBuffer(int nextWrite) { return localCache.remaining() >= nextWrite; } private void flushBuffer(byte[] buffer, int offset, int length) { if (firstPart) { client.putObject( new PutObjectRequest( uri.bucket(), uri.name(), new ByteArrayInputStream(buffer, offset, length))); firstPart = false; } else { client.appendObject( uri.bucket(), uri.name(), new ByteArrayInputStream(buffer, offset, length)); } } /** Pos of the file */ @Override public long getPos() { return pos; } /** Write cached bytes if present. */ @Override public void flush() { if (localCache.remaining() < localCache.capacity()) { localCache.flip(); flushBuffer(localCache.array(), localCache.arrayOffset(), localCache.remaining()); localCache.clear(); } } /** Trigger flush() when closing stream. */ @Override public void close() { flush(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy