/*
 * Copyright 2002-2018 the original author or authors.
 *
 * 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 org.springframework.http.codec.protobuf;

import java.io.IOException;
import java.util.Arrays;

import com.google.protobuf.Message;
import org.junit.Test;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.test.StepVerifier;

import org.springframework.core.ResolvableType;
import org.springframework.core.codec.AbstractDecoderTestCase;
import org.springframework.core.codec.DecodingException;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.core.io.buffer.DataBufferUtils;
import org.springframework.http.MediaType;
import org.springframework.protobuf.Msg;
import org.springframework.protobuf.SecondMsg;
import org.springframework.util.MimeType;

import static java.util.Collections.emptyMap;
import static org.junit.Assert.*;
import static org.springframework.core.ResolvableType.forClass;
import static org.springframework.core.io.buffer.DataBufferUtils.release;

/**
 * Unit tests for {@link ProtobufDecoder}.
 *
 * @author Sebastien Deleuze
 */
public class ProtobufDecoderTests extends AbstractDecoderTestCase<ProtobufDecoder> {

	private final static MimeType PROTOBUF_MIME_TYPE = new MimeType("application", "x-protobuf");

	private final SecondMsg secondMsg = SecondMsg.newBuilder().setBlah(123).build();

	private final Msg testMsg1 = Msg.newBuilder().setFoo("Foo").setBlah(secondMsg).build();

	private final SecondMsg secondMsg2 = SecondMsg.newBuilder().setBlah(456).build();

	private final Msg testMsg2 = Msg.newBuilder().setFoo("Bar").setBlah(secondMsg2).build();

	public ProtobufDecoderTests() {
		super(new ProtobufDecoder());
	}


	@Test(expected = IllegalArgumentException.class)
	public void extensionRegistryNull() {
		new ProtobufDecoder(null);
	}

	@Override
	@Test
	public void canDecode() {
		assertTrue(this.decoder.canDecode(forClass(Msg.class), null));
		assertTrue(this.decoder.canDecode(forClass(Msg.class), PROTOBUF_MIME_TYPE));
		assertTrue(this.decoder.canDecode(forClass(Msg.class), MediaType.APPLICATION_OCTET_STREAM));
		assertFalse(this.decoder.canDecode(forClass(Msg.class), MediaType.APPLICATION_JSON));
		assertFalse(this.decoder.canDecode(forClass(Object.class), PROTOBUF_MIME_TYPE));
	}

	@Override
	@Test
	public void decodeToMono() {
		Mono<DataBuffer> input = dataBuffer(this.testMsg1);

		testDecodeToMonoAll(input, Msg.class, step -> step
				.expectNext(this.testMsg1)
				.verifyComplete());
	}

	@Test
	public void decodeChunksToMono() {
		byte[] full = this.testMsg1.toByteArray();
		byte[] chunk1 = Arrays.copyOfRange(full, 0, full.length / 2);
		byte[] chunk2 = Arrays.copyOfRange(full, chunk1.length, full.length);

		Flux<DataBuffer> input = Flux.just(chunk1, chunk2)
				.flatMap(bytes -> Mono.defer(() -> {
					DataBuffer dataBuffer = this.bufferFactory.allocateBuffer(bytes.length);
					dataBuffer.write(bytes);
					return Mono.just(dataBuffer);
				}));

		testDecodeToMono(input, Msg.class, step -> step
				.expectNext(this.testMsg1)
				.verifyComplete());
	}

	@Override
	@Test
	public void decode() {
		Flux<DataBuffer> input = Flux.just(this.testMsg1, this.testMsg2)
				.flatMap(msg -> Mono.defer(() -> {
					DataBuffer buffer = this.bufferFactory.allocateBuffer();
					try {
						msg.writeDelimitedTo(buffer.asOutputStream());
						return Mono.just(buffer);
					}
					catch (IOException e) {
						release(buffer);
						return Mono.error(e);
					}
				}));

		testDecodeAll(input, Msg.class, step -> step
				.expectNext(this.testMsg1)
				.expectNext(this.testMsg2)
				.verifyComplete());
	}

	@Test
	public void decodeSplitChunks() {


		Flux<DataBuffer> input = Flux.just(this.testMsg1, this.testMsg2)
				.flatMap(msg -> Mono.defer(() -> {
					DataBuffer buffer = this.bufferFactory.allocateBuffer();
					try {
						msg.writeDelimitedTo(buffer.asOutputStream());
						return Mono.just(buffer);
					}
					catch (IOException e) {
						release(buffer);
						return Mono.error(e);
					}
				}))
				.flatMap(buffer -> {
					int len = buffer.readableByteCount() / 2;
					Flux<DataBuffer> result = Flux.just(
							DataBufferUtils.retain(buffer.slice(0, len)),
							DataBufferUtils
									.retain(buffer.slice(len, buffer.readableByteCount() - len))
					);
					release(buffer);
					return result;
				});

		testDecode(input, Msg.class, step -> step
				.expectNext(this.testMsg1)
				.expectNext(this.testMsg2)
				.verifyComplete());
	}

	@Test  // SPR-17429
	public void decodeSplitMessageSize() {
		this.decoder.setMaxMessageSize(100009);
		StringBuilder builder = new StringBuilder();
		for (int i = 0; i < 10000; i++) {
			builder.append("azertyuiop");
		}
		Msg bigMessage = Msg.newBuilder().setFoo(builder.toString()).setBlah(secondMsg2).build();

		Flux<DataBuffer> input = Flux.just(bigMessage, bigMessage)
				.flatMap(msg -> Mono.defer(() -> {
					DataBuffer buffer = this.bufferFactory.allocateBuffer();
					try {
						msg.writeDelimitedTo(buffer.asOutputStream());
						return Mono.just(buffer);
					}
					catch (IOException e) {
						release(buffer);
						return Mono.error(e);
					}
				}))
				.flatMap(buffer -> {
					int len = 2;
					Flux<DataBuffer> result = Flux.just(
							DataBufferUtils.retain(buffer.slice(0, len)),
							DataBufferUtils
									.retain(buffer.slice(len, buffer.readableByteCount() - len))
					);
					release(buffer);
					return result;
				});

		testDecode(input, Msg.class, step -> step
				.expectNext(bigMessage)
				.expectNext(bigMessage)
				.verifyComplete());
	}

	@Test
	public void decodeMergedChunks() throws IOException {
		DataBuffer buffer = this.bufferFactory.allocateBuffer();
		this.testMsg1.writeDelimitedTo(buffer.asOutputStream());
		this.testMsg1.writeDelimitedTo(buffer.asOutputStream());

		ResolvableType elementType = forClass(Msg.class);
		Flux<Message> messages = this.decoder.decode(Mono.just(buffer), elementType, null, emptyMap());

		StepVerifier.create(messages)
				.expectNext(testMsg1)
				.expectNext(testMsg1)
				.verifyComplete();
	}

	@Test
	public void exceedMaxSize() {
		this.decoder.setMaxMessageSize(1);
		Mono<DataBuffer> input = dataBuffer(this.testMsg1);

		testDecode(input, Msg.class, step -> step
				.verifyError(DecodingException.class));
	}

	private Mono<DataBuffer> dataBuffer(Msg msg) {
		return Mono.defer(() -> {
			byte[] bytes = msg.toByteArray();
			DataBuffer buffer = this.bufferFactory.allocateBuffer(bytes.length);
			buffer.write(bytes);
			return Mono.just(buffer);
		});
	}


}