/*
 * Copyright 2002-2019 Drew Noakes and contributors
 *
 *    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.
 *
 * More information about this project is available at:
 *
 *    https://drewnoakes.com/code/exif/
 *    https://github.com/drewnoakes/metadata-extractor
 */
package com.drew.metadata;

import com.drew.lang.Rational;
import com.drew.metadata.exif.ExifIFD0Directory;
import com.drew.metadata.exif.ExifSubIFDDirectory;
import org.junit.Before;
import org.junit.Test;

import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.TimeZone;

import static org.junit.Assert.*;

/**
 * @author Drew Noakes https://drewnoakes.com
 */
public class DirectoryTest
{
    // TODO write tests to validate type conversions from all underlying types

    private Directory _directory;

    @Before public void setup()
    {
        _directory = new MockDirectory();
    }

    @Test public void testSetAndGetMultipleTagsInSingleDirectory() throws Exception
    {
        _directory.setString(ExifSubIFDDirectory.TAG_APERTURE, "TAG_APERTURE");
        _directory.setString(ExifSubIFDDirectory.TAG_BATTERY_LEVEL, "TAG_BATTERY_LEVEL");
        assertEquals("TAG_APERTURE", _directory.getString(ExifSubIFDDirectory.TAG_APERTURE));
        assertEquals("TAG_BATTERY_LEVEL", _directory.getString(ExifSubIFDDirectory.TAG_BATTERY_LEVEL));
    }

    @Test public void testSetSameTagMultipleTimesOverwritesValue() throws Exception
    {
        _directory.setInt(ExifSubIFDDirectory.TAG_APERTURE, 1);
        _directory.setInt(ExifSubIFDDirectory.TAG_APERTURE, 2);
        assertEquals(2, _directory.getInt(ExifSubIFDDirectory.TAG_APERTURE));
    }

    @Test
    public void testUnderlyingInt() throws Exception
    {
        int value = 123;
        int tagType = 321;
        _directory.setInt(tagType, value);

        assertEquals(value, _directory.getInt(tagType));
        assertEquals(Integer.valueOf(value), _directory.getInteger(tagType));
        assertEquals((float)value, _directory.getFloat(tagType), 0.00001);
        assertEquals((double)value, _directory.getDouble(tagType), 0.00001);
        assertEquals((long)value, _directory.getLong(tagType));
        assertEquals(Integer.toString(value), _directory.getString(tagType));
        assertEquals(new Rational(value, 1), _directory.getRational(tagType));
        assertArrayEquals(new int[]{value}, _directory.getIntArray(tagType));
        assertArrayEquals(new byte[]{(byte)value}, _directory.getByteArray(tagType));
    }

    @Test public void testSetAndGetIntArray() throws Exception
    {
        int[] inputValues = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
        int tagType = 123;
        _directory.setIntArray(tagType, inputValues);
        int[] outputValues = _directory.getIntArray(tagType);
        assertNotNull(outputValues);
        assertEquals(inputValues.length, outputValues.length);
        for (int i = 0; i < inputValues.length; i++) {
            int inputValue = inputValues[i];
            int outputValue = outputValues[i];
            assertEquals(inputValue, outputValue);
        }
        assertArrayEquals(inputValues, _directory.getIntArray(tagType));
        StringBuilder outputString = new StringBuilder();
        for (int i = 0; i < inputValues.length; i++) {
            int inputValue = inputValues[i];
            if (i > 0) {
                outputString.append(' ');
            }
            outputString.append(inputValue);
        }
        assertEquals(outputString.toString(), _directory.getString(tagType));
    }

    @Test
    public void testSetStringAndGetDate() throws Exception
    {
        String date1 = "2002:01:30 23:59:59";
        String date2 = "2002:01:30 23:59";
        String date3 = "2002-01-30 23:59:59";
        String date4 = "2002-01-30 23:59";
        String date5 = "2002-01-30T23:59:59.099-08:00";
        String date6 = "2002-01-30T23:59:59.099";
        String date7 = "2002-01-30T23:59:59-08:00";
        String date8 = "2002-01-30T23:59:59";
        String date9 = "2002-01-30T23:59-08:00";
        String date10 = "2002-01-30T23:59";
        String date11 = "2002-01-30";
        String date12 = "2002-01";
        String date13 = "2002";
        _directory.setString(1, date1);
        _directory.setString(2, date2);
        _directory.setString(3, date3);
        _directory.setString(4, date4);
        _directory.setString(5, date5);
        _directory.setString(6, date6);
        _directory.setString(7, date7);
        _directory.setString(8, date8);
        _directory.setString(9, date9);
        _directory.setString(10, date10);
        _directory.setString(11, date11);
        _directory.setString(12, date12);
        _directory.setString(13, date13);
        assertEquals(date1, _directory.getString(1));

            // Don't use default timezone
        TimeZone gmt = TimeZone.getTimeZone("GMT");
        GregorianCalendar gc = new GregorianCalendar(gmt);
            // clear millis to 0 or test will fail
        gc.setTimeInMillis(0);
        gc.set(2002, GregorianCalendar.JANUARY, 30, 23, 59, 59);
        assertEquals(gc.getTime(), _directory.getDate(1, null));

        gc.set(2002, GregorianCalendar.JANUARY, 30, 23, 59, 0);
        assertEquals(gc.getTime(), _directory.getDate(2, null));

            // Use specific timezone
        TimeZone pst = TimeZone.getTimeZone("PST");
        gc = new GregorianCalendar(pst);
        gc.setTimeInMillis(0);

        gc.set(2002, GregorianCalendar.JANUARY, 30, 23, 59, 59);
        assertEquals(gc.getTime(), _directory.getDate(3, pst));

        gc.set(2002, GregorianCalendar.JANUARY, 30, 23, 59, 0);
        assertEquals(gc.getTime(), _directory.getDate(4, pst));

        gc.set(2002, GregorianCalendar.JANUARY, 30, 23, 59, 59);
        gc.set(Calendar.MILLISECOND, 99);
        assertEquals(gc.getTime(), _directory.getDate(5, null));
        assertEquals(gc.getTime(), _directory.getDate(5, gmt));
        assertEquals(gc.getTime(), _directory.getDate(6, pst));

        assertEquals(gc.getTime(), _directory.getDate(5, "011", null));
        assertEquals(gc.getTime(), _directory.getDate(6, "011", pst));
        assertEquals(gc.getTime(), _directory.getDate(7, "099", null));
        assertEquals(gc.getTime(), _directory.getDate(8, "099", pst));

        gc.set(Calendar.MILLISECOND, 0);
        assertEquals(gc.getTime(), _directory.getDate(7, null));
        assertEquals(gc.getTime(), _directory.getDate(7, gmt));
        assertEquals(gc.getTime(), _directory.getDate(8, pst));

        gc.set(2002, GregorianCalendar.JANUARY, 30, 23, 59, 0);
        assertEquals(gc.getTime(), _directory.getDate(9, null));
        assertEquals(gc.getTime(), _directory.getDate(9, gmt));
        assertEquals(gc.getTime(), _directory.getDate(10, pst));

        gc = new GregorianCalendar(gmt);
        gc.setTimeInMillis(0);

        gc.set(2002, GregorianCalendar.JANUARY, 30, 0, 0, 0);
        assertEquals(gc.getTime(), _directory.getDate(11, null));

        gc.set(2002, GregorianCalendar.JANUARY, 1, 0, 0, 0);
        assertEquals(gc.getTime(), _directory.getDate(12, null));

        gc.set(2002, GregorianCalendar.JANUARY, 1, 0, 0, 0);
        assertEquals(gc.getTime(), _directory.getDate(13, null));
    }

    @Test
    public void testSetIntArrayGetByteArray() throws Exception
    {
        int[] ints = {1, 2, 3, 4, 5};
        _directory.setIntArray(1, ints);

        byte[] bytes = _directory.getByteArray(1);
        assertNotNull(bytes);
        assertEquals(ints.length, bytes.length);
        assertEquals(1, bytes[0]);
    }

    @Test
    public void testSetStringGetInt() throws Exception
    {
        byte[] bytes = { 0x01, 0x02, 0x03 };
        _directory.setString(1, new String(bytes));
        assertEquals(0x010203, _directory.getInt(1));
    }

    @Test
    public void testContainsTag() throws Exception
    {
        assertFalse(_directory.containsTag(ExifSubIFDDirectory.TAG_APERTURE));
        _directory.setString(ExifSubIFDDirectory.TAG_APERTURE, "Tag Value");
        assertTrue(_directory.containsTag(ExifSubIFDDirectory.TAG_APERTURE));
    }

    @Test
    public void testGetNonExistentTagIsNullForAllTypes() throws Exception
    {
        assertNull(_directory.getString(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getInteger(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getDoubleObject(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getFloatObject(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getByteArray(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getDate(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getIntArray(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getLongObject(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getObject(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getRational(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getRationalArray(ExifSubIFDDirectory.TAG_APERTURE));
        assertNull(_directory.getStringArray(ExifSubIFDDirectory.TAG_APERTURE));
    }

    @Test
    public void testToString()
    {
        Directory directory = new ExifIFD0Directory();
        assertEquals("Exif IFD0 Directory (0 tags)", directory.toString());
        directory.setString(1, "Tag 1");
        assertEquals("Exif IFD0 Directory (1 tag)", directory.toString());
        directory.setString(2, "Tag 2");
        assertEquals("Exif IFD0 Directory (2 tags)", directory.toString());
    }
}