/*
 * Copyright 2002-2019 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.web.servlet.resource;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.zip.GZIPOutputStream;

import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Test;

import org.springframework.cache.Cache;
import org.springframework.cache.concurrent.ConcurrentMapCache;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.FileSystemResource;
import org.springframework.core.io.Resource;
import org.springframework.http.HttpHeaders;
import org.springframework.mock.web.test.MockHttpServletRequest;
import org.springframework.util.FileCopyUtils;

import static org.junit.Assert.*;

/**
 * Unit tests for {@link EncodedResourceResolver}.
 *
 * @author Jeremy Grelle
 * @author Rossen Stoyanchev
 */
public class EncodedResourceResolverTests {

	private ResourceResolverChain resolver;

	private List<Resource> locations;

	private Cache cache;


	@BeforeClass
	public static void createGzippedResources() throws IOException {
		createGzippedFile("/js/foo.js");
		createGzippedFile("foo.css");
	}

	static void createGzippedFile(String filePath) throws IOException {
		Resource location = new ClassPathResource("test/", EncodedResourceResolverTests.class);
		Resource resource = new FileSystemResource(location.createRelative(filePath).getFile());

		Path gzFilePath = Paths.get(resource.getFile().getAbsolutePath() + ".gz");
		Files.deleteIfExists(gzFilePath);

		File gzFile = Files.createFile(gzFilePath).toFile();
		GZIPOutputStream out = new GZIPOutputStream(new FileOutputStream(gzFile));
		FileCopyUtils.copy(resource.getInputStream(), out);
		gzFile.deleteOnExit();
	}


	@Before
	public void setup() {
		this.cache = new ConcurrentMapCache("resourceCache");

		VersionResourceResolver versionResolver = new VersionResourceResolver();
		versionResolver.setStrategyMap(Collections.singletonMap("/**", new ContentVersionStrategy()));

		List<ResourceResolver> resolvers = new ArrayList<>();
		resolvers.add(new CachingResourceResolver(this.cache));
		resolvers.add(new EncodedResourceResolver());
		resolvers.add(versionResolver);
		resolvers.add(new PathResourceResolver());
		this.resolver = new DefaultResourceResolverChain(resolvers);

		this.locations = new ArrayList<>();
		this.locations.add(new ClassPathResource("test/", getClass()));
		this.locations.add(new ClassPathResource("testalternatepath/", getClass()));
	}


	@Test
	public void resolveGzipped() {
		String file = "js/foo.js";
		MockHttpServletRequest request = new MockHttpServletRequest();
		request.addHeader("Accept-Encoding", "gzip");
		Resource actual = this.resolver.resolveResource(request, file, this.locations);

		assertEquals(getResource(file + ".gz").getDescription(), actual.getDescription());
		assertEquals(getResource(file).getFilename(), actual.getFilename());

		assertTrue(actual instanceof HttpResource);
		HttpHeaders headers = ((HttpResource) actual).getResponseHeaders();
		assertEquals("gzip", headers.getFirst(HttpHeaders.CONTENT_ENCODING));
		assertEquals("Accept-Encoding", headers.getFirst(HttpHeaders.VARY));
	}

	@Test
	public void resolveGzippedWithVersion() {
		String file = "foo-e36d2e05253c6c7085a91522ce43a0b4.css";
		MockHttpServletRequest request = new MockHttpServletRequest();
		request.addHeader("Accept-Encoding", "gzip");
		Resource resolved = this.resolver.resolveResource(request, file, this.locations);

		assertEquals(getResource("foo.css.gz").getDescription(), resolved.getDescription());
		assertEquals(getResource("foo.css").getFilename(), resolved.getFilename());
		assertTrue(resolved instanceof HttpResource);
	}

	@Test
	public void resolveFromCacheWithEncodingVariants() {
		// 1. Resolve, and cache .gz variant
		String file = "js/foo.js";
		MockHttpServletRequest request = new MockHttpServletRequest("GET", "/js/foo.js");
		request.addHeader("Accept-Encoding", "gzip");
		Resource resolved = this.resolver.resolveResource(request, file, this.locations);

		assertEquals(getResource(file + ".gz").getDescription(), resolved.getDescription());
		assertEquals(getResource(file).getFilename(), resolved.getFilename());
		assertTrue(resolved instanceof HttpResource);

		// 2. Resolve unencoded resource
		request = new MockHttpServletRequest("GET", "/js/foo.js");
		resolved = this.resolver.resolveResource(request, file, this.locations);

		assertEquals(getResource(file).getDescription(), resolved.getDescription());
		assertEquals(getResource(file).getFilename(), resolved.getFilename());
		assertFalse(resolved instanceof HttpResource);
	}

	@Test  // SPR-13149
	public void resolveWithNullRequest() {
		String file = "js/foo.js";
		Resource resolved = this.resolver.resolveResource(null, file, this.locations);

		assertEquals(getResource(file).getDescription(), resolved.getDescription());
		assertEquals(getResource(file).getFilename(), resolved.getFilename());
	}

	private Resource getResource(String filePath) {
		return new ClassPathResource("test/" + filePath, getClass());
	}

}