// Copyright (c) 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include "packager/base/files/file_util.h" #include "packager/media/base/container_names.h" #include "packager/media/test/test_data_util.h" namespace shaka { namespace media { // Using a macros to simplify tests. Since EXPECT_EQ outputs the second argument // as a string when it fails, this lets the output identify what item actually // failed. #define VERIFY(buffer, name) \ EXPECT_EQ(name, \ DetermineContainer(reinterpret_cast(buffer), \ sizeof(buffer))) // Test that small buffers are handled correctly. TEST(ContainerNamesTest, CheckSmallBuffer) { // Empty buffer. char buffer[1]; // ([0] not allowed on win) VERIFY(buffer, CONTAINER_UNKNOWN); // Try a simple SRT file. char buffer1[] = "1\n" "00:03:23,550 --> 00:03:24,375\n" "You always had a hard time finding your place in this world.\n" "\n" "2\n" "00:03:24,476 --> 00:03:25,175\n" "What are you talking about?\n"; VERIFY(buffer1, CONTAINER_SRT); // HLS has it's own loop. char buffer2[] = "#EXTM3U" "some other random stuff" "#EXT-X-MEDIA-SEQUENCE:"; VERIFY(buffer2, CONTAINER_HLS); // Try a large buffer all zeros. char buffer3[4096]; memset(buffer3, 0, sizeof(buffer3)); VERIFY(buffer3, CONTAINER_UNKNOWN); // Reuse buffer, but all \n this time. memset(buffer3, '\n', sizeof(buffer3)); VERIFY(buffer3, CONTAINER_UNKNOWN); } #define BYTE_ORDER_MARK "\xef\xbb\xbf" // Note that the comparisons need at least 12 bytes, so make sure the buffer is // at least that size. const char kAmrBuffer[12] = "#!AMR"; uint8_t kAsfBuffer[] = {0x30, 0x26, 0xb2, 0x75, 0x8e, 0x66, 0xcf, 0x11, 0xa6, 0xd9, 0x00, 0xaa, 0x00, 0x62, 0xce, 0x6c}; const char kAss1Buffer[] = "[Script Info]"; const char kAss2Buffer[] = BYTE_ORDER_MARK "[Script Info]"; uint8_t kCafBuffer[] = { 'c', 'a', 'f', 'f', 0, 1, 0, 0, 'd', 'e', 's', 'c', 0, 0, 0, 0, 0, 0, 0, 32, 64, 229, 136, 128, 0, 0, 0, 0, 'a', 'a', 'c', ' ', 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 4, 0, 0, 0, 0, 2, 0, 0, 0, 0}; const char kDtshdBuffer[12] = "DTSHDHDR"; const char kDxaBuffer[16] = "DEXA"; const char kFlacBuffer[12] = "fLaC"; uint8_t kFlvBuffer[12] = {'F', 'L', 'V', 0, 0, 0, 0, 1, 0, 0, 0, 0}; uint8_t kIrcamBuffer[] = {0x64, 0xa3, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1}; const char kRm1Buffer[12] = ".RMF\0\0"; const char kRm2Buffer[12] = ".ra\xfd"; uint8_t kWtvBuffer[] = {0xb7, 0xd8, 0x00, 0x20, 0x37, 0x49, 0xda, 0x11, 0xa6, 0x4e, 0x00, 0x07, 0xe9, 0x5e, 0xad, 0x8d}; uint8_t kBug263073Buffer[] = { 0x00, 0x00, 0x00, 0x18, 0x66, 0x74, 0x79, 0x70, 0x6d, 0x70, 0x34, 0x32, 0x00, 0x00, 0x00, 0x00, 0x69, 0x73, 0x6f, 0x6d, 0x6d, 0x70, 0x34, 0x32, 0x00, 0x00, 0x00, 0x01, 0x6d, 0x64, 0x61, 0x74, 0x00, 0x00, 0x00, 0x00, 0xaa, 0x2e, 0x22, 0xcf, 0x00, 0x00, 0x00, 0x37, 0x67, 0x64, 0x00, 0x28, 0xac, 0x2c, 0xa4, 0x01, 0xe0, 0x08, 0x9f, 0x97, 0x01, 0x52, 0x02, 0x02, 0x02, 0x80, 0x00, 0x01}; TEST(ContainerNamesTest, FromFormatName) { EXPECT_EQ(CONTAINER_WEBM, DetermineContainerFromFormatName("webm")); EXPECT_EQ(CONTAINER_WEBM, DetermineContainerFromFormatName("WeBm")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("m4a")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("m4v")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("M4v")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("m4s")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("mov")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("mp4")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFormatName("Mp4")); EXPECT_EQ(CONTAINER_MPEG2TS, DetermineContainerFromFormatName("ts")); EXPECT_EQ(CONTAINER_MPEG2TS, DetermineContainerFromFormatName("mpeg2ts")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFormatName("cat")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFormatName("amp4")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFormatName(" mp4")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFormatName("")); } TEST(ContainerNamesTest, FromFileName) { EXPECT_EQ(CONTAINER_WEBM, DetermineContainerFromFileName("test.webm")); EXPECT_EQ(CONTAINER_WEBM, DetermineContainerFromFileName("another.wEbM")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFileName("test.m4a")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFileName("file.m4v")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFileName("a file .m4V")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFileName("segment.m4s")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFileName("2_more-files.mp4")); EXPECT_EQ(CONTAINER_MOV, DetermineContainerFromFileName("foo.bar.MP4")); EXPECT_EQ(CONTAINER_MPEG2TS, DetermineContainerFromFileName("a.ts")); EXPECT_EQ(CONTAINER_MPEG2TS, DetermineContainerFromFileName("a.TS")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFileName("a_bad.gif")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFileName("a bad.m4v-")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFileName("a.m4v.")); EXPECT_EQ(CONTAINER_UNKNOWN, DetermineContainerFromFileName("")); } // Test that containers that start with fixed strings are handled correctly. // This is to verify that the TAG matches the first 4 characters of the string. TEST(ContainerNamesTest, CheckFixedStrings) { VERIFY(kAmrBuffer, CONTAINER_AMR); VERIFY(kAsfBuffer, CONTAINER_ASF); VERIFY(kAss1Buffer, CONTAINER_ASS); VERIFY(kAss2Buffer, CONTAINER_ASS); VERIFY(kCafBuffer, CONTAINER_CAF); VERIFY(kDtshdBuffer, CONTAINER_DTSHD); VERIFY(kDxaBuffer, CONTAINER_DXA); VERIFY(kFlacBuffer, CONTAINER_FLAC); VERIFY(kFlvBuffer, CONTAINER_FLV); VERIFY(kIrcamBuffer, CONTAINER_IRCAM); VERIFY(kRm1Buffer, CONTAINER_RM); VERIFY(kRm2Buffer, CONTAINER_RM); VERIFY(kWtvBuffer, CONTAINER_WTV); VERIFY(kBug263073Buffer, CONTAINER_MOV); } // Determine the container type of a specified file. void TestFile(MediaContainerName expected, const base::FilePath& filename) { char buffer[8192]; // Windows implementation of ReadFile fails if file smaller than desired size, // so use file length if file less than 8192 bytes (http://crbug.com/243885). int read_size = sizeof(buffer); int64_t actual_size; if (base::GetFileSize(filename, &actual_size) && actual_size < read_size) read_size = actual_size; int read = base::ReadFile(filename, buffer, read_size); ASSERT_GT(read, 0) << filename.value(); // Now verify the type. EXPECT_EQ(expected, DetermineContainer(reinterpret_cast(buffer), read)) << "Failure with file " << filename.value(); } TEST(ContainerNamesTest, Ttml) { // One of the actual TTMLs from the TTML spec page. const char kTtml[] = "\n" "\n" " \n" "
\n" "

\n" " Some subtitle.\n" "

\n" "
\n" " \n" "
\n"; EXPECT_EQ(CONTAINER_TTML, DetermineContainer(reinterpret_cast(kTtml), arraysize(kTtml))); } TEST(ContainerNamesTest, WebVtt) { const char kWebVtt[] = "WEBVTT\n" "\n" "00:1.000 --> 00:2.000\n" "Subtitle"; EXPECT_EQ(CONTAINER_WEBVTT, DetermineContainer(reinterpret_cast(kWebVtt), arraysize(kWebVtt))); const uint8_t kUtf8ByteOrderMark[] = {0xef, 0xbb, 0xbf}; std::vector webvtt_with_utf8_byte_order_mark( kUtf8ByteOrderMark, kUtf8ByteOrderMark + arraysize(kUtf8ByteOrderMark)); webvtt_with_utf8_byte_order_mark.insert( webvtt_with_utf8_byte_order_mark.end(), kWebVtt, kWebVtt + arraysize(kWebVtt)); EXPECT_EQ(CONTAINER_WEBVTT, DetermineContainer( webvtt_with_utf8_byte_order_mark.data(), static_cast(webvtt_with_utf8_byte_order_mark.size()))); } TEST(ContainerNamesTest, FileCheckOGG) { TestFile(CONTAINER_OGG, GetTestDataFilePath("bear.ogv")); TestFile(CONTAINER_OGG, GetTestDataFilePath("9ch.ogg")); } TEST(ContainerNamesTest, FileCheckWAV) { TestFile(CONTAINER_WAV, GetTestDataFilePath("4ch.wav")); } TEST(ContainerNamesTest, FileCheckMOV) { TestFile(CONTAINER_MOV, GetTestDataFilePath("bear-640x360.mp4")); } TEST(ContainerNamesTest, FileCheckWEBM) { TestFile(CONTAINER_WEBM, GetTestDataFilePath("bear-640x360.webm")); TestFile(CONTAINER_WEBM, GetTestDataFilePath("no_streams.webm")); } TEST(ContainerNamesTest, FileCheckMP3) { TestFile(CONTAINER_MP3, GetTestDataFilePath("id3_test.mp3")); } TEST(ContainerNamesTest, FileCheckAC3) { TestFile(CONTAINER_AC3, GetTestDataFilePath("bear.ac3")); } TEST(ContainerNamesTest, FileCheckAAC) { TestFile(CONTAINER_AAC, GetTestDataFilePath("bear.adts")); } TEST(ContainerNamesTest, FileCheckAIFF) { TestFile(CONTAINER_AIFF, GetTestDataFilePath("bear.aiff")); } TEST(ContainerNamesTest, FileCheckASF) { TestFile(CONTAINER_ASF, GetTestDataFilePath("bear.asf")); } TEST(ContainerNamesTest, FileCheckAVI) { TestFile(CONTAINER_AVI, GetTestDataFilePath("bear.avi")); } TEST(ContainerNamesTest, FileCheckEAC3) { TestFile(CONTAINER_EAC3, GetTestDataFilePath("bear.eac3")); } TEST(ContainerNamesTest, FileCheckFLAC) { TestFile(CONTAINER_FLAC, GetTestDataFilePath("bear.flac")); } TEST(ContainerNamesTest, FileCheckFLV) { TestFile(CONTAINER_FLV, GetTestDataFilePath("bear.flv")); } TEST(ContainerNamesTest, FileCheckH261) { TestFile(CONTAINER_H261, GetTestDataFilePath("bear.h261")); } TEST(ContainerNamesTest, FileCheckH263) { TestFile(CONTAINER_H263, GetTestDataFilePath("bear.h263")); } TEST(ContainerNamesTest, FileCheckMJPEG) { TestFile(CONTAINER_MJPEG, GetTestDataFilePath("bear.mjpeg")); } TEST(ContainerNamesTest, FileCheckMPEG2PS) { TestFile(CONTAINER_MPEG2PS, GetTestDataFilePath("bear.mpeg")); } TEST(ContainerNamesTest, FileCheckMPEG2TS) { TestFile(CONTAINER_MPEG2TS, GetTestDataFilePath("bear.m2ts")); } TEST(ContainerNamesTest, FileCheckRM) { TestFile(CONTAINER_RM, GetTestDataFilePath("bear.rm")); } TEST(ContainerNamesTest, FileCheckSWF) { TestFile(CONTAINER_SWF, GetTestDataFilePath("bear.swf")); } // Try a few non containers. TEST(ContainerNamesTest, FileCheckUNKNOWN) { TestFile(CONTAINER_UNKNOWN, GetTestDataFilePath("ten_byte_file")); TestFile(CONTAINER_UNKNOWN, GetTestDataFilePath("README")); } } // namespace media } // namespace shaka