/*
 * 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.flink.runtime.state.filesystem;

import org.apache.flink.core.fs.FSDataInputStream;
import org.apache.flink.core.fs.FSDataOutputStream;
import org.apache.flink.core.fs.FileSystem;
import org.apache.flink.core.fs.Path;
import org.apache.flink.core.fs.local.LocalDataOutputStream;
import org.apache.flink.core.fs.local.LocalFileSystem;
import org.apache.flink.core.testutils.CheckedThread;
import org.apache.flink.core.testutils.OneShotLatch;
import org.apache.flink.util.TestLogger;
import org.apache.flink.util.function.FunctionWithException;

import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;

import java.io.EOFException;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
import java.util.Collection;
import java.util.Random;
import java.util.UUID;

import static org.junit.Assert.assertArrayEquals;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify;

/**
 * Abstract base class for tests against checkpointing streams.
 */
@RunWith(Parameterized.class)
public class CheckpointStateOutputStreamTest extends TestLogger {

	@Rule
	public final TemporaryFolder tmp = new TemporaryFolder();

	public enum CheckpointStateOutputStreamType {
		FileBasedState, FsCheckpointMetaData
	}

	@Parameterized.Parameters
	public static Collection<CheckpointStateOutputStreamType> getCheckpointStateOutputStreamType() {
		return Arrays.asList(CheckpointStateOutputStreamType.values());
	}

	@Parameterized.Parameter
	public CheckpointStateOutputStreamType stateOutputStreamType;

	// ------------------------------------------------------------------------
	//  Tests
	// ------------------------------------------------------------------------

	/**
	 * Validates that even empty streams create a file and a file state handle.
	 */
	@Test
	public void testEmptyState() throws Exception {
		final FileSystem fs = FileSystem.getLocalFileSystem();
		final Path folder = baseFolder();
		final String fileName = "myFileName";
		final Path filePath = new Path(folder, fileName);

		final FileStateHandle handle;
		try (FSDataOutputStream stream = createTestStream(fs, folder, fileName)) {
			handle = closeAndGetResult(stream);
		}

		// must have created a handle
		assertNotNull(handle);
		assertEquals(filePath, handle.getFilePath());

		// the pointer path should exist as a directory
		assertTrue(fs.exists(handle.getFilePath()));
		assertFalse(fs.getFileStatus(filePath).isDir());

		// the contents should be empty
		try (FSDataInputStream in = handle.openInputStream()) {
			assertEquals(-1, in.read());
		}
	}

	/**
	 * Simple write and read test
	 */
	@Test
	public void testWriteAndRead() throws Exception {
		final FileSystem fs = FileSystem.getLocalFileSystem();
		final Path folder = baseFolder();
		final String fileName = "fooBarName";

		final Random rnd = new Random();
		final byte[] data = new byte[1694523];

		// write the data (mixed single byte writes and array writes)
		final FileStateHandle handle;
		try (FSDataOutputStream stream = createTestStream(fs, folder, fileName)) {
			for (int i = 0; i < data.length; ) {
				if (rnd.nextBoolean()) {
					stream.write(data[i++]);
				} else {
					int len = rnd.nextInt(Math.min(data.length - i, 32));
					stream.write(data, i, len);
					i += len;
				}
			}
			handle = closeAndGetResult(stream);
		}

		// (1) stream from handle must hold the contents
		try (FSDataInputStream in = handle.openInputStream()) {
			byte[] buffer = new byte[data.length];
			readFully(in, buffer);
			assertArrayEquals(data, buffer);
		}

		// (2) the pointer must point to a file with that contents
		try (FSDataInputStream in = fs.open(handle.getFilePath())) {
			byte[] buffer = new byte[data.length];
			readFully(in, buffer);
			assertArrayEquals(data, buffer);
		}
	}

	/**
	 * Tests that the underlying stream file is deleted upon calling close.
	 */
	@Test
	public void testCleanupWhenClosingStream() throws IOException {
		final FileSystem fs = FileSystem.getLocalFileSystem();
		final Path folder = new Path(tmp.newFolder().toURI());
		final String fileName = "nonCreativeTestFileName";
		final Path path = new Path(folder, fileName);

		// write some test data and close the stream
		try (FSDataOutputStream stream = createTestStream(fs, folder, fileName)) {
			Random rnd = new Random();
			for (int i = 0; i < rnd.nextInt(1000); i++) {
				stream.write(rnd.nextInt(100));
			}
			assertTrue(fs.exists(path));
		}

		assertFalse(fs.exists(path));
	}

	/**
	 * Tests that the underlying stream file is deleted if the closeAndGetHandle method fails.
	 */
	@Test
	public void testCleanupWhenFailingCloseAndGetHandle() throws IOException {
		final Path folder = new Path(tmp.newFolder().toURI());
		final String fileName = "test_name";
		final Path filePath = new Path(folder, fileName);

		final FileSystem fs = spy(new TestFs((path) -> new FailingCloseStream(new File(path.getPath()))));

		FSDataOutputStream stream = createTestStream(fs, folder, fileName);
		stream.write(new byte[] {1, 2, 3, 4, 5});

		try {
			closeAndGetResult(stream);
			fail("Expected IOException");
		}
		catch (IOException ignored) {
			// expected exception
		}

		verify(fs).delete(filePath, false);
	}

	/**
	 * This test validates that a close operation can happen even while a 'closeAndGetHandle()'
	 * call is in progress.
	 * <p>
	 * <p>That behavior is essential for fast cancellation (concurrent cleanup).
	 */
	@Test
	public void testCloseDoesNotLock() throws Exception {
		final Path folder = new Path(tmp.newFolder().toURI());
		final String fileName = "this-is-ignored-anyways.file";

		final FileSystem fileSystem = spy(new TestFs((path) -> new BlockerStream()));

		final FSDataOutputStream checkpointStream =
			createTestStream(fileSystem, folder, fileName);

		final OneShotLatch sync = new OneShotLatch();

		final CheckedThread thread = new CheckedThread() {

			@Override
			public void go() throws Exception {
				sync.trigger();
				// that call should now block, because it accesses the position
				closeAndGetResult(checkpointStream);
			}
		};
		thread.start();

		sync.await();
		checkpointStream.close();

		// the thread may or may not fail, that depends on the thread race
		// it is not important for this test, important is that the thread does not freeze/lock up
		try {
			thread.sync();
		} catch (IOException ignored) {}
	}

	/**
	 * Creates a new test stream instance.
	 */
	private FSDataOutputStream createTestStream(
		FileSystem fs,
		Path dir,
		String fileName) throws IOException {
		switch (stateOutputStreamType) {
			case FileBasedState:
				return new FileBasedStateOutputStream(fs, new Path(dir, fileName));
			case FsCheckpointMetaData:
				Path fullPath = new Path(dir, fileName);
				return new FsCheckpointMetadataOutputStream(fs, fullPath, dir);
			default:
				throw new IllegalStateException("Unsupported checkpoint stream output type.");
		}
	}

	/**
	 * Closes the stream successfully and returns a FileStateHandle to the result.
	 */
	private FileStateHandle closeAndGetResult(FSDataOutputStream stream) throws IOException {
		switch (stateOutputStreamType) {
			case FileBasedState:
				return ((FileBasedStateOutputStream) stream).closeAndGetHandle();
			case FsCheckpointMetaData:
				return ((FsCheckpointMetadataOutputStream) stream).closeAndFinalizeCheckpoint().getMetadataHandle();
			default:
				throw new IllegalStateException("Unsupported checkpoint stream output type.");
		}
	}

	// ------------------------------------------------------------------------
	//  utilities
	// ------------------------------------------------------------------------

	private Path baseFolder() throws Exception {
		return new Path(new File(tmp.newFolder(), UUID.randomUUID().toString()).toURI());
	}

	private static void readFully(InputStream in, byte[] buffer) throws IOException {
		int pos = 0;
		int remaining = buffer.length;

		while (remaining > 0) {
			int read = in.read(buffer, pos, remaining);
			if (read == -1) {
				throw new EOFException();
			}

			pos += read;
			remaining -= read;
		}
	}

	private static class BlockerStream extends FSDataOutputStream {

		private final OneShotLatch blocker = new OneShotLatch();

		@Override
		public long getPos() throws IOException {
			block();
			return 0L;
		}

		@Override
		public void write(int b) throws IOException {
			block();
		}

		@Override
		public void flush() throws IOException {
			block();
		}

		@Override
		public void sync() throws IOException {
			block();
		}

		@Override
		public void close() throws IOException {
			blocker.trigger();
		}

		private void block() throws IOException {
			try {
				blocker.await();
			} catch (InterruptedException e) {
				throw new IOException("interrupted");
			}
			throw new IOException("closed");
		}
	}

	// ------------------------------------------------------------------------

	private static class FailingCloseStream extends LocalDataOutputStream {

		FailingCloseStream(File file) throws IOException {
			super(file);
		}

		@Override
		public void close() throws IOException {
			throw new IOException();
		}
	}

	private static class TestFs extends LocalFileSystem {

		private final FunctionWithException<Path, FSDataOutputStream, IOException> streamFactory;

		TestFs(FunctionWithException<Path, FSDataOutputStream, IOException> streamFactory) {
			this.streamFactory = streamFactory;
		}

		@Override
		public FSDataOutputStream create(Path filePath, WriteMode overwrite) throws IOException {
			return streamFactory.apply(filePath);
		}
	}

}