/* GStreamer unit test for MPEG-DASH * * Copyright (c) <2015> YouView TV Ltd * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include "../../ext/dash/gstmpdparser.c" #undef GST_CAT_DEFAULT #include GST_DEBUG_CATEGORY (gst_dash_demux_debug); /* * compute the number of milliseconds contained in a duration value specified by * year, month, day, hour, minute, second, millisecond * * This function must use the same conversion algorithm implemented in * gst_mpdparser_get_xml_prop_duration from gstmpdparser.c file. */ static guint64 duration_to_ms (guint year, guint month, guint day, guint hour, guint minute, guint second, guint millisecond) { guint64 days = (guint64) year * 365 + (guint64) month * 30 + day; guint64 hours = days * 24 + hour; guint64 minutes = hours * 60 + minute; guint64 seconds = minutes * 60 + second; guint64 ms = seconds * 1000 + millisecond; return ms; } /* * Test to ensure a simple mpd file successfully parses. * */ GST_START_TEST (dash_mpdparser_validsimplempd) { const gchar *xml = "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* check that unset elements with default values are properly configured */ assert_equals_int (mpdclient->mpd_node->type, GST_MPD_FILE_TYPE_STATIC); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing the MPD attributes. * */ GST_START_TEST (dash_mpdparser_mpd) { GstDateTime *availabilityStartTime; GstDateTime *availabilityEndTime; const gchar *xml = "" ""; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); assert_equals_string (mpdclient->mpd_node->default_namespace, "urn:mpeg:dash:schema:mpd:2011"); assert_equals_string (mpdclient->mpd_node->namespace_xsi, "TestNamespaceXSI"); assert_equals_string (mpdclient->mpd_node->namespace_ext, "TestNamespaceEXT"); assert_equals_string (mpdclient->mpd_node->schemaLocation, "TestSchemaLocation"); assert_equals_string (mpdclient->mpd_node->id, "testId"); assert_equals_int (mpdclient->mpd_node->type, GST_MPD_FILE_TYPE_STATIC); availabilityStartTime = mpdclient->mpd_node->availabilityStartTime; assert_equals_int (gst_date_time_get_year (availabilityStartTime), 2015); assert_equals_int (gst_date_time_get_month (availabilityStartTime), 3); assert_equals_int (gst_date_time_get_day (availabilityStartTime), 24); assert_equals_int (gst_date_time_get_hour (availabilityStartTime), 1); assert_equals_int (gst_date_time_get_minute (availabilityStartTime), 10); assert_equals_int (gst_date_time_get_second (availabilityStartTime), 50); availabilityEndTime = mpdclient->mpd_node->availabilityEndTime; assert_equals_int (gst_date_time_get_year (availabilityEndTime), 2015); assert_equals_int (gst_date_time_get_month (availabilityEndTime), 3); assert_equals_int (gst_date_time_get_day (availabilityEndTime), 24); assert_equals_int (gst_date_time_get_hour (availabilityEndTime), 1); assert_equals_int (gst_date_time_get_minute (availabilityEndTime), 10); assert_equals_int (gst_date_time_get_second (availabilityEndTime), 50); assert_equals_int64 (mpdclient->mpd_node->mediaPresentationDuration, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (mpdclient->mpd_node->minimumUpdatePeriod, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (mpdclient->mpd_node->minBufferTime, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (mpdclient->mpd_node->timeShiftBufferDepth, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (mpdclient->mpd_node->suggestedPresentationDelay, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (mpdclient->mpd_node->maxSegmentDuration, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (mpdclient->mpd_node->maxSubsegmentDuration, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing the ProgramInformation attributes * */ GST_START_TEST (dash_mpdparser_programInformation) { GstProgramInformationNode *program; const gchar *xml = "" "" " " " TestTitle" " TestSource" " TestCopyright" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); program = (GstProgramInformationNode *) mpdclient->mpd_node->ProgramInfo->data; assert_equals_string (program->lang, "en"); assert_equals_string (program->moreInformationURL, "TestMoreInformationUrl"); assert_equals_string (program->Title, "TestTitle"); assert_equals_string (program->Source, "TestSource"); assert_equals_string (program->Copyright, "TestCopyright"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing the BaseURL attributes * */ GST_START_TEST (dash_mpdparser_baseURL) { GstBaseURL *baseURL; const gchar *xml = "" "" " TestBaseURL"; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); baseURL = (GstBaseURL *) mpdclient->mpd_node->BaseURLs->data; assert_equals_string (baseURL->baseURL, "TestBaseURL"); assert_equals_string (baseURL->serviceLocation, "TestServiceLocation"); assert_equals_string (baseURL->byteRange, "TestByteRange"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing the Location attributes * */ GST_START_TEST (dash_mpdparser_location) { const gchar *location; const gchar *xml = "" "" " TestLocation"; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); location = (gchar *) mpdclient->mpd_node->Locations->data; assert_equals_string (location, "TestLocation"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Metrics attributes * */ GST_START_TEST (dash_mpdparser_metrics) { GstMetricsNode *metricsNode; const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); metricsNode = (GstMetricsNode *) mpdclient->mpd_node->Metrics->data; assert_equals_string (metricsNode->metrics, "TestMetric"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Metrics Range attributes * */ GST_START_TEST (dash_mpdparser_metrics_range) { GstMetricsNode *metricsNode; GstMetricsRangeNode *metricsRangeNode; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); metricsNode = (GstMetricsNode *) mpdclient->mpd_node->Metrics->data; assert_equals_pointer (metricsNode->metrics, NULL); metricsRangeNode = (GstMetricsRangeNode *) metricsNode->MetricsRanges->data; assert_equals_int64 (metricsRangeNode->starttime, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 500)); assert_equals_int64 (metricsRangeNode->duration, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 123)); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Metrics Reporting attributes * */ GST_START_TEST (dash_mpdparser_metrics_reporting) { GstMetricsNode *metricsNode; const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); metricsNode = (GstMetricsNode *) mpdclient->mpd_node->Metrics->data; assert_equals_pointer (metricsNode->metrics, NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period attributes * */ GST_START_TEST (dash_mpdparser_period) { GstPeriodNode *periodNode; const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; assert_equals_string (periodNode->id, "TestId"); assert_equals_int64 (periodNode->start, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 123)); assert_equals_int64 (periodNode->duration, (gint64) duration_to_ms (0, 1, 2, 12, 10, 20, 765)); assert_equals_int (periodNode->bitstreamSwitching, 1); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period baseURL attributes * */ GST_START_TEST (dash_mpdparser_period_baseURL) { GstPeriodNode *periodNode; GstBaseURL *baseURL; const gchar *xml = "" "" " " " TestBaseURL" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; baseURL = (GstBaseURL *) periodNode->BaseURLs->data; assert_equals_string (baseURL->baseURL, "TestBaseURL"); assert_equals_string (baseURL->serviceLocation, "TestServiceLocation"); assert_equals_string (baseURL->byteRange, "TestByteRange"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentBase attributes * */ GST_START_TEST (dash_mpdparser_period_segmentBase) { GstPeriodNode *periodNode; GstSegmentBaseType *segmentBase; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentBase = periodNode->SegmentBase; assert_equals_uint64 (segmentBase->timescale, 123456); assert_equals_uint64 (segmentBase->presentationTimeOffset, 123456789); assert_equals_uint64 (segmentBase->indexRange->first_byte_pos, 100); assert_equals_uint64 (segmentBase->indexRange->last_byte_pos, 200); assert_equals_int (segmentBase->indexRangeExact, 1); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentBase Initialization attributes * */ GST_START_TEST (dash_mpdparser_period_segmentBase_initialization) { GstPeriodNode *periodNode; GstSegmentBaseType *segmentBase; GstURLType *initialization; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentBase = periodNode->SegmentBase; initialization = segmentBase->Initialization; assert_equals_string (initialization->sourceURL, "TestSourceURL"); assert_equals_uint64 (initialization->range->first_byte_pos, 100); assert_equals_uint64 (initialization->range->last_byte_pos, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentBase RepresentationIndex attributes * */ GST_START_TEST (dash_mpdparser_period_segmentBase_representationIndex) { GstPeriodNode *periodNode; GstSegmentBaseType *segmentBase; GstURLType *representationIndex; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentBase = periodNode->SegmentBase; representationIndex = segmentBase->RepresentationIndex; assert_equals_string (representationIndex->sourceURL, "TestSourceURL"); assert_equals_uint64 (representationIndex->range->first_byte_pos, 100); assert_equals_uint64 (representationIndex->range->last_byte_pos, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList attributes * */ GST_START_TEST (dash_mpdparser_period_segmentList) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; fail_if (segmentList == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList MultipleSegmentBaseType attributes * */ GST_START_TEST (dash_mpdparser_period_segmentList_multipleSegmentBaseType) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; GstMultSegmentBaseType *multSegBaseType; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; multSegBaseType = segmentList->MultSegBaseType; assert_equals_uint64 (multSegBaseType->duration, 10); assert_equals_uint64 (multSegBaseType->startNumber, 11); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList MultipleSegmentBaseType SegmentBaseType * attributes */ GST_START_TEST (dash_mpdparser_period_segmentList_multipleSegmentBaseType_segmentBaseType) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; GstMultSegmentBaseType *multSegBaseType; GstSegmentBaseType *segBaseType; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; multSegBaseType = segmentList->MultSegBaseType; segBaseType = multSegBaseType->SegBaseType; assert_equals_uint64 (segBaseType->timescale, 10); assert_equals_uint64 (segBaseType->presentationTimeOffset, 11); assert_equals_uint64 (segBaseType->indexRange->first_byte_pos, 20); assert_equals_uint64 (segBaseType->indexRange->last_byte_pos, 21); assert_equals_int (segBaseType->indexRangeExact, FALSE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList MultipleSegmentBaseType SegmentTimeline * attributes */ GST_START_TEST (dash_mpdparser_period_segmentList_multipleSegmentBaseType_segmentTimeline) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; GstMultSegmentBaseType *multSegBaseType; GstSegmentTimelineNode *segmentTimeline; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; multSegBaseType = segmentList->MultSegBaseType; segmentTimeline = multSegBaseType->SegmentTimeline; fail_if (segmentTimeline == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList MultipleSegmentBaseType SegmentTimeline S * attributes */ GST_START_TEST (dash_mpdparser_period_segmentList_multipleSegmentBaseType_segmentTimeline_s) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; GstMultSegmentBaseType *multSegBaseType; GstSegmentTimelineNode *segmentTimeline; GstSNode *sNode; const gchar *xml = "" "" " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; multSegBaseType = segmentList->MultSegBaseType; segmentTimeline = multSegBaseType->SegmentTimeline; sNode = (GstSNode *) g_queue_peek_head (&segmentTimeline->S); assert_equals_uint64 (sNode->t, 1); assert_equals_uint64 (sNode->d, 2); assert_equals_uint64 (sNode->r, 3); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList MultipleSegmentBaseType BitstreamSwitching * attributes */ GST_START_TEST (dash_mpdparser_period_segmentList_multipleSegmentBaseType_bitstreamSwitching) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; GstMultSegmentBaseType *multSegBaseType; GstURLType *bitstreamSwitching; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; multSegBaseType = segmentList->MultSegBaseType; bitstreamSwitching = multSegBaseType->BitstreamSwitching; assert_equals_string (bitstreamSwitching->sourceURL, "TestSourceURL"); assert_equals_uint64 (bitstreamSwitching->range->first_byte_pos, 100); assert_equals_uint64 (bitstreamSwitching->range->last_byte_pos, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentList SegmentURL attributes * */ GST_START_TEST (dash_mpdparser_period_segmentList_segmentURL) { GstPeriodNode *periodNode; GstSegmentListNode *segmentList; GstSegmentURLNode *segmentURL; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentList = periodNode->SegmentList; segmentURL = (GstSegmentURLNode *) segmentList->SegmentURL->data; assert_equals_string (segmentURL->media, "TestMedia"); assert_equals_uint64 (segmentURL->mediaRange->first_byte_pos, 100); assert_equals_uint64 (segmentURL->mediaRange->last_byte_pos, 200); assert_equals_string (segmentURL->index, "TestIndex"); assert_equals_uint64 (segmentURL->indexRange->first_byte_pos, 300); assert_equals_uint64 (segmentURL->indexRange->last_byte_pos, 400); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentTemplate attributes * */ GST_START_TEST (dash_mpdparser_period_segmentTemplate) { GstPeriodNode *periodNode; GstSegmentTemplateNode *segmentTemplate; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentTemplate = periodNode->SegmentTemplate; assert_equals_string (segmentTemplate->media, "TestMedia"); assert_equals_string (segmentTemplate->index, "TestIndex"); assert_equals_string (segmentTemplate->initialization, "TestInitialization"); assert_equals_string (segmentTemplate->bitstreamSwitching, "TestBitstreamSwitching"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentTemplate MultipleSegmentBaseType attributes * */ GST_START_TEST (dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType) { GstPeriodNode *periodNode; GstSegmentTemplateNode *segmentTemplate; GstMultSegmentBaseType *multSegBaseType; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentTemplate = periodNode->SegmentTemplate; multSegBaseType = segmentTemplate->MultSegBaseType; assert_equals_uint64 (multSegBaseType->duration, 10); assert_equals_uint64 (multSegBaseType->startNumber, 11); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentTemplate MultipleSegmentBaseType SegmentBaseType * attributes */ GST_START_TEST (dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_segmentBaseType) { GstPeriodNode *periodNode; GstSegmentTemplateNode *segmentTemplate; GstMultSegmentBaseType *multSegBaseType; GstSegmentBaseType *segBaseType; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentTemplate = periodNode->SegmentTemplate; multSegBaseType = segmentTemplate->MultSegBaseType; segBaseType = multSegBaseType->SegBaseType; assert_equals_uint64 (segBaseType->timescale, 123456); assert_equals_uint64 (segBaseType->presentationTimeOffset, 123456789); assert_equals_uint64 (segBaseType->indexRange->first_byte_pos, 100); assert_equals_uint64 (segBaseType->indexRange->last_byte_pos, 200); assert_equals_int (segBaseType->indexRangeExact, TRUE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentTemplate MultipleSegmentBaseType SegmentTimeline * attributes */ GST_START_TEST (dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_segmentTimeline) { GstPeriodNode *periodNode; GstSegmentTemplateNode *segmentTemplate; GstMultSegmentBaseType *multSegBaseType; GstSegmentTimelineNode *segmentTimeline; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentTemplate = periodNode->SegmentTemplate; multSegBaseType = segmentTemplate->MultSegBaseType; segmentTimeline = (GstSegmentTimelineNode *) multSegBaseType->SegmentTimeline; fail_if (segmentTimeline == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentTemplate MultipleSegmentBaseType SegmentTimeline * S attributes */ GST_START_TEST (dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_segmentTimeline_s) { GstPeriodNode *periodNode; GstSegmentTemplateNode *segmentTemplate; GstMultSegmentBaseType *multSegBaseType; GstSegmentTimelineNode *segmentTimeline; GstSNode *sNode; const gchar *xml = "" "" " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentTemplate = periodNode->SegmentTemplate; multSegBaseType = segmentTemplate->MultSegBaseType; segmentTimeline = (GstSegmentTimelineNode *) multSegBaseType->SegmentTimeline; sNode = (GstSNode *) g_queue_peek_head (&segmentTimeline->S); assert_equals_uint64 (sNode->t, 1); assert_equals_uint64 (sNode->d, 2); assert_equals_uint64 (sNode->r, 3); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period SegmentTemplate MultipleSegmentBaseType * BitstreamSwitching attributes */ GST_START_TEST (dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_bitstreamSwitching) { GstPeriodNode *periodNode; GstSegmentTemplateNode *segmentTemplate; GstMultSegmentBaseType *multSegBaseType; GstURLType *bitstreamSwitching; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; segmentTemplate = periodNode->SegmentTemplate; multSegBaseType = segmentTemplate->MultSegBaseType; bitstreamSwitching = multSegBaseType->BitstreamSwitching; assert_equals_string (bitstreamSwitching->sourceURL, "TestSourceURL"); assert_equals_uint64 (bitstreamSwitching->range->first_byte_pos, 100); assert_equals_uint64 (bitstreamSwitching->range->last_byte_pos, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; assert_equals_uint64 (adaptationSet->id, 7); assert_equals_uint64 (adaptationSet->group, 8); assert_equals_string (adaptationSet->lang, "en"); assert_equals_string (adaptationSet->contentType, "TestContentType"); assert_equals_uint64 (adaptationSet->par->num, 4); assert_equals_uint64 (adaptationSet->par->den, 3); assert_equals_uint64 (adaptationSet->minBandwidth, 100); assert_equals_uint64 (adaptationSet->maxBandwidth, 200); assert_equals_uint64 (adaptationSet->minWidth, 1000); assert_equals_uint64 (adaptationSet->maxWidth, 2000); assert_equals_uint64 (adaptationSet->minHeight, 1100); assert_equals_uint64 (adaptationSet->maxHeight, 2100); assert_equals_uint64 (adaptationSet->minFrameRate->num, 25); assert_equals_uint64 (adaptationSet->minFrameRate->den, 123); assert_equals_uint64 (adaptationSet->maxFrameRate->num, 26); assert_equals_uint64 (adaptationSet->maxFrameRate->den, 1); assert_equals_int (adaptationSet->segmentAlignment->flag, 1); assert_equals_uint64 (adaptationSet->segmentAlignment->value, 2); assert_equals_int (adaptationSet->subsegmentAlignment->flag, 0); assert_equals_uint64 (adaptationSet->subsegmentAlignment->value, 0); assert_equals_int (adaptationSet->subsegmentStartsWithSAP, 6); assert_equals_int (adaptationSet->bitstreamSwitching, 0); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet RepresentationBase attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representationBase) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationBaseType *representationBase; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representationBase = adaptationSet->RepresentationBase; assert_equals_string (representationBase->profiles, "TestProfiles"); assert_equals_uint64 (representationBase->width, 100); assert_equals_uint64 (representationBase->height, 200); assert_equals_uint64 (representationBase->sar->num, 10); assert_equals_uint64 (representationBase->sar->den, 20); assert_equals_uint64 (representationBase->frameRate->num, 30); assert_equals_uint64 (representationBase->frameRate->den, 40); assert_equals_string (representationBase->audioSamplingRate, "TestAudioSamplingRate"); assert_equals_string (representationBase->mimeType, "TestMimeType"); assert_equals_string (representationBase->segmentProfiles, "TestSegmentProfiles"); assert_equals_string (representationBase->codecs, "TestCodecs"); assert_equals_float (representationBase->maximumSAPPeriod, 3.4); assert_equals_int (representationBase->startWithSAP, GST_SAP_TYPE_0); assert_equals_float (representationBase->maxPlayoutRate, 1.2); assert_equals_float (representationBase->codingDependency, 0); assert_equals_string (representationBase->scanType, "progressive"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet RepresentationBase FramePacking attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representationBase_framePacking) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationBaseType *representationBase; GstDescriptorType *framePacking; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representationBase = adaptationSet->RepresentationBase; framePacking = (GstDescriptorType *) representationBase->FramePacking->data; assert_equals_string (framePacking->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (framePacking->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet RepresentationBase * AudioChannelConfiguration attributes */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representationBase_audioChannelConfiguration) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationBaseType *representationBase; GstDescriptorType *audioChannelConfiguration; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representationBase = adaptationSet->RepresentationBase; audioChannelConfiguration = (GstDescriptorType *) representationBase->AudioChannelConfiguration->data; assert_equals_string (audioChannelConfiguration->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (audioChannelConfiguration->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet RepresentationBase ContentProtection * attributes */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representationBase_contentProtection) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationBaseType *representationBase; GstDescriptorType *contentProtection; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representationBase = adaptationSet->RepresentationBase; contentProtection = (GstDescriptorType *) representationBase->ContentProtection->data; assert_equals_string (contentProtection->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (contentProtection->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Accessibility attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_accessibility) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstDescriptorType *accessibility; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; accessibility = (GstDescriptorType *) adaptationSet->Accessibility->data; assert_equals_string (accessibility->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (accessibility->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Role attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_role) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstDescriptorType *role; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; role = (GstDescriptorType *) adaptationSet->Role->data; assert_equals_string (role->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (role->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Rating attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_rating) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstDescriptorType *rating; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; rating = (GstDescriptorType *) adaptationSet->Rating->data; assert_equals_string (rating->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (rating->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Viewpoint attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_viewpoint) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstDescriptorType *viewpoint; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; viewpoint = (GstDescriptorType *) adaptationSet->Viewpoint->data; assert_equals_string (viewpoint->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (viewpoint->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet ContentComponent attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_contentComponent) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstContentComponentNode *contentComponent; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; contentComponent = (GstContentComponentNode *) adaptationSet->ContentComponents->data; assert_equals_uint64 (contentComponent->id, 1); assert_equals_string (contentComponent->lang, "en"); assert_equals_string (contentComponent->contentType, "TestContentType"); assert_equals_uint64 (contentComponent->par->num, 10); assert_equals_uint64 (contentComponent->par->den, 20); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet ContentComponent Accessibility attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_contentComponent_accessibility) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstContentComponentNode *contentComponent; GstDescriptorType *accessibility; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; contentComponent = (GstContentComponentNode *) adaptationSet->ContentComponents->data; accessibility = (GstDescriptorType *) contentComponent->Accessibility->data; assert_equals_string (accessibility->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (accessibility->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet ContentComponent Role attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_contentComponent_role) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstContentComponentNode *contentComponent; GstDescriptorType *role; const gchar *xml = "" "" " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; contentComponent = (GstContentComponentNode *) adaptationSet->ContentComponents->data; role = (GstDescriptorType *) contentComponent->Role->data; assert_equals_string (role->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (role->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet ContentComponent Rating attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_contentComponent_rating) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstContentComponentNode *contentComponent; GstDescriptorType *rating; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; contentComponent = (GstContentComponentNode *) adaptationSet->ContentComponents->data; rating = (GstDescriptorType *) contentComponent->Rating->data; assert_equals_string (rating->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (rating->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet ContentComponent Viewpoint attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_contentComponent_viewpoint) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstContentComponentNode *contentComponent; GstDescriptorType *viewpoint; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; contentComponent = (GstContentComponentNode *) adaptationSet->ContentComponents->data; viewpoint = (GstDescriptorType *) contentComponent->Viewpoint->data; assert_equals_string (viewpoint->schemeIdUri, "TestSchemeIdUri"); assert_equals_string (viewpoint->value, "TestValue"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet BaseURL attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_baseURL) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstBaseURL *baseURL; const gchar *xml = "" "" " " " " " TestBaseURL" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; baseURL = (GstBaseURL *) adaptationSet->BaseURLs->data; assert_equals_string (baseURL->baseURL, "TestBaseURL"); assert_equals_string (baseURL->serviceLocation, "TestServiceLocation"); assert_equals_string (baseURL->byteRange, "TestByteRange"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet SegmentBase attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_segmentBase) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstSegmentBaseType *segmentBase; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; segmentBase = adaptationSet->SegmentBase; assert_equals_uint64 (segmentBase->timescale, 123456); assert_equals_uint64 (segmentBase->presentationTimeOffset, 123456789); assert_equals_uint64 (segmentBase->indexRange->first_byte_pos, 100); assert_equals_uint64 (segmentBase->indexRange->last_byte_pos, 200); assert_equals_int (segmentBase->indexRangeExact, TRUE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet SegmentBase Initialization attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_segmentBase_initialization) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstSegmentBaseType *segmentBase; GstURLType *initialization; const gchar *xml = "" "" " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; segmentBase = adaptationSet->SegmentBase; initialization = segmentBase->Initialization; assert_equals_string (initialization->sourceURL, "TestSourceURL"); assert_equals_uint64 (initialization->range->first_byte_pos, 100); assert_equals_uint64 (initialization->range->last_byte_pos, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet SegmentBase RepresentationIndex attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_segmentBase_representationIndex) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstSegmentBaseType *segmentBase; GstURLType *representationIndex; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; segmentBase = adaptationSet->SegmentBase; representationIndex = segmentBase->RepresentationIndex; assert_equals_string (representationIndex->sourceURL, "TestSourceURL"); assert_equals_uint64 (representationIndex->range->first_byte_pos, 100); assert_equals_uint64 (representationIndex->range->last_byte_pos, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet SegmentList attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_segmentList) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstSegmentListNode *segmentList; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; segmentList = adaptationSet->SegmentList; fail_if (segmentList == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet SegmentTemplate attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_segmentTemplate) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstSegmentTemplateNode *segmentTemplate; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; segmentTemplate = adaptationSet->SegmentTemplate; assert_equals_string (segmentTemplate->media, "TestMedia"); assert_equals_string (segmentTemplate->index, "TestIndex"); assert_equals_string (segmentTemplate->initialization, "TestInitialization"); assert_equals_string (segmentTemplate->bitstreamSwitching, "TestBitstreamSwitching"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; assert_equals_string (representation->id, "Test Id"); assert_equals_uint64 (representation->bandwidth, 100); assert_equals_uint64 (representation->qualityRanking, 200); assert_equals_string (representation->dependencyId[0], "one"); assert_equals_string (representation->dependencyId[1], "two"); assert_equals_string (representation->dependencyId[2], "three"); assert_equals_pointer (representation->dependencyId[3], NULL); assert_equals_pointer (representation->mediaStreamStructureId[0], NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation RepresentationBaseType attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_representationBase) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstRepresentationBaseType *representationBase; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; representationBase = (GstRepresentationBaseType *) representation->RepresentationBase; fail_if (representationBase == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation BaseURL attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_baseURL) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstBaseURL *baseURL; const gchar *xml = "" "" " " " " " " " TestBaseURL" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; baseURL = (GstBaseURL *) representation->BaseURLs->data; assert_equals_string (baseURL->baseURL, "TestBaseURL"); assert_equals_string (baseURL->serviceLocation, "TestServiceLocation"); assert_equals_string (baseURL->byteRange, "TestByteRange"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation SubRepresentation attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_subRepresentation) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstSubRepresentationNode *subRepresentation; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; subRepresentation = (GstSubRepresentationNode *) representation->SubRepresentations->data; assert_equals_uint64 (subRepresentation->level, 100); assert_equals_uint64 (subRepresentation->size, 3); assert_equals_uint64 (subRepresentation->dependencyLevel[0], 1); assert_equals_uint64 (subRepresentation->dependencyLevel[1], 2); assert_equals_uint64 (subRepresentation->dependencyLevel[2], 3); assert_equals_uint64 (subRepresentation->bandwidth, 200); assert_equals_string (subRepresentation->contentComponent[0], "content1"); assert_equals_string (subRepresentation->contentComponent[1], "content2"); assert_equals_pointer (subRepresentation->contentComponent[2], NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation SubRepresentation * RepresentationBase attributes */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_subRepresentation_representationBase) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstSubRepresentationNode *subRepresentation; GstRepresentationBaseType *representationBase; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; subRepresentation = (GstSubRepresentationNode *) representation->SubRepresentations->data; representationBase = (GstRepresentationBaseType *) subRepresentation->RepresentationBase; fail_if (representationBase == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation SegmentBase attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_segmentBase) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstSegmentBaseType *segmentBase; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; segmentBase = representation->SegmentBase; fail_if (segmentBase == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation SegmentList attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_segmentList) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstSegmentListNode *segmentList; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; segmentList = representation->SegmentList; fail_if (segmentList == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period AdaptationSet Representation SegmentTemplate attributes * */ GST_START_TEST (dash_mpdparser_period_adaptationSet_representation_segmentTemplate) { GstPeriodNode *periodNode; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; GstSegmentTemplateNode *segmentTemplate; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; segmentTemplate = representation->SegmentTemplate; fail_if (segmentTemplate == NULL); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing Period Subset attributes * */ GST_START_TEST (dash_mpdparser_period_subset) { GstPeriodNode *periodNode; GstSubsetNode *subset; const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; subset = (GstSubsetNode *) periodNode->Subsets->data; assert_equals_uint64 (subset->size, 3); assert_equals_uint64 (subset->contains[0], 1); assert_equals_uint64 (subset->contains[1], 2); assert_equals_uint64 (subset->contains[2], 3); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing the type property: value "dynamic" * */ GST_START_TEST (dash_mpdparser_type_dynamic) { gboolean isLive; const gchar *xml = "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); isLive = gst_mpd_client_is_live (mpdclient); assert_equals_int (isLive, 1); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Validate gst_mpdparser_build_URL_from_template function * */ GST_START_TEST (dash_mpdparser_template_parsing) { const gchar *url_template; const gchar *id = "TestId"; guint number = 7; guint bandwidth = 2500; guint64 time = 100; gchar *result; url_template = "TestMedia$Bandwidth$$$test"; result = gst_mpdparser_build_URL_from_template (url_template, id, number, bandwidth, time); assert_equals_string (result, "TestMedia2500$test"); g_free (result); } GST_END_TEST; /* * Test handling isoff ondemand profile * */ GST_START_TEST (dash_mpdparser_isoff_ondemand_profile) { gboolean hasOnDemandProfile; const gchar *xml = "" ""; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); hasOnDemandProfile = gst_mpd_client_has_isoff_ondemand_profile (mpdclient); assert_equals_int (hasOnDemandProfile, 1); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling GstDateTime * */ GST_START_TEST (dash_mpdparser_GstDateTime) { gint64 delta; GstDateTime *time1; GstDateTime *time2; GstDateTime *time3; GDateTime *g_time2; GDateTime *g_time3; time1 = gst_date_time_new_from_iso8601_string ("2012-06-23T23:30:59Z"); time2 = gst_date_time_new_from_iso8601_string ("2012-06-23T23:31:00Z"); delta = gst_mpd_client_calculate_time_difference (time1, time2); assert_equals_int64 (delta, 1 * GST_SECOND); time3 = gst_mpd_client_add_time_difference (time1, GST_TIME_AS_USECONDS (delta)); /* convert to GDateTime in order to compare time2 and time 3 */ g_time2 = gst_date_time_to_g_date_time (time2); g_time3 = gst_date_time_to_g_date_time (time3); fail_if (g_date_time_compare (g_time2, g_time3) != 0); gst_date_time_unref (time1); gst_date_time_unref (time2); gst_date_time_unref (time3); g_date_time_unref (g_time2); g_date_time_unref (g_time3); } GST_END_TEST; /* * Test media presentation setup * */ GST_START_TEST (dash_mpdparser_setup_media_presentation) { const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test setting a stream * */ GST_START_TEST (dash_mpdparser_setup_streaming) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the first adaptation set of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); adapt_set = (GstAdaptationSetNode *) adaptationSets->data; fail_if (adapt_set == NULL); /* setup streaming from the adaptation set */ ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling Period selection * */ GST_START_TEST (dash_mpdparser_period_selection) { const gchar *periodName; guint periodIndex; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* period_idx should be 0 and we should have no active periods */ assert_equals_uint64 (mpdclient->period_idx, 0); fail_unless (mpdclient->periods == NULL); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* check the periods */ fail_unless (mpdclient->periods != NULL); periodName = gst_mpd_client_get_period_id (mpdclient); assert_equals_string (periodName, "Period0"); ret = gst_mpd_client_set_period_index (mpdclient, 1); assert_equals_int (ret, TRUE); periodName = gst_mpd_client_get_period_id (mpdclient); assert_equals_string (periodName, "Period1"); ret = gst_mpd_client_set_period_index (mpdclient, 2); assert_equals_int (ret, TRUE); periodName = gst_mpd_client_get_period_id (mpdclient); assert_equals_string (periodName, "Period2"); ret = gst_mpd_client_has_next_period (mpdclient); assert_equals_int (ret, FALSE); ret = gst_mpd_client_has_previous_period (mpdclient); assert_equals_int (ret, TRUE); ret = gst_mpd_client_set_period_index (mpdclient, 0); assert_equals_int (ret, TRUE); ret = gst_mpd_client_has_next_period (mpdclient); assert_equals_int (ret, TRUE); ret = gst_mpd_client_has_previous_period (mpdclient); assert_equals_int (ret, FALSE); ret = gst_mpd_client_set_period_id (mpdclient, "Period1"); assert_equals_int (ret, TRUE); periodIndex = gst_mpd_client_get_period_index (mpdclient); assert_equals_uint64 (periodIndex, 1); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling Period selection based on time * */ GST_START_TEST (dash_mpdparser_get_period_at_time) { guint periodIndex; GstDateTime *time; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* request period for a time before availabilityStartTime, expect period index 0 */ time = gst_date_time_new_from_iso8601_string ("2015-03-23T23:30:59Z"); periodIndex = gst_mpd_client_get_period_index_at_time (mpdclient, time); gst_date_time_unref (time); assert_equals_int (periodIndex, 0); /* request period for a time from period 0 */ time = gst_date_time_new_from_iso8601_string ("2015-03-24T23:30:59Z"); periodIndex = gst_mpd_client_get_period_index_at_time (mpdclient, time); gst_date_time_unref (time); assert_equals_int (periodIndex, 0); /* request period for a time from period 1 */ time = gst_date_time_new_from_iso8601_string ("2015-03-25T1:1:1Z"); periodIndex = gst_mpd_client_get_period_index_at_time (mpdclient, time); gst_date_time_unref (time); assert_equals_int (periodIndex, 1); /* request period for a time from period 2 */ time = gst_date_time_new_from_iso8601_string ("2015-03-25T1:3:3Z"); periodIndex = gst_mpd_client_get_period_index_at_time (mpdclient, time); gst_date_time_unref (time); assert_equals_int (periodIndex, 2); /* request period for a time after mediaPresentationDuration, expect period index G_MAXUINT */ time = gst_date_time_new_from_iso8601_string ("2015-03-25T1:4:3Z"); periodIndex = gst_mpd_client_get_period_index_at_time (mpdclient, time); gst_date_time_unref (time); assert_equals_int (periodIndex, G_MAXUINT); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling Adaptation sets * */ GST_START_TEST (dash_mpdparser_adaptationSet_handling) { const gchar *periodName; guint adaptation_sets_count; GList *adaptationSets; guint count = 0; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* period0 has 1 adaptation set */ fail_unless (mpdclient->periods != NULL); periodName = gst_mpd_client_get_period_id (mpdclient); assert_equals_string (periodName, "Period0"); adaptation_sets_count = gst_mpdparser_get_nb_adaptationSet (mpdclient); assert_equals_int (adaptation_sets_count, 1); /* period1 has 2 adaptation set */ ret = gst_mpd_client_set_period_id (mpdclient, "Period1"); assert_equals_int (ret, TRUE); adaptation_sets_count = gst_mpdparser_get_nb_adaptationSet (mpdclient); assert_equals_int (adaptation_sets_count, 2); /* check the id for the 2 adaptation sets from period 1 */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); for (GList * it = adaptationSets; it; it = g_list_next (it)) { GstAdaptationSetNode *adapt_set; adapt_set = (GstAdaptationSetNode *) it->data; fail_if (adapt_set == NULL); assert_equals_int (adapt_set->id, 10 + count); count++; } gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling Representation selection * */ GST_START_TEST (dash_mpdparser_representation_selection) { GList *adaptationSets; GstAdaptationSetNode *adaptationSetNode; GList *representations; gint represendationIndex; const gchar *xml = "" "" " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); adaptationSetNode = adaptationSets->data; fail_if (adaptationSetNode == NULL); assert_equals_int (adaptationSetNode->id, 1); representations = adaptationSetNode->Representations; fail_if (representations == NULL); represendationIndex = gst_mpdparser_get_rep_idx_with_min_bandwidth (representations); assert_equals_int (represendationIndex, 1); represendationIndex = gst_mpdparser_get_rep_idx_with_max_bandwidth (representations, 0); assert_equals_int (represendationIndex, 1); represendationIndex = gst_mpdparser_get_rep_idx_with_max_bandwidth (representations, 100000); assert_equals_int (represendationIndex, -1); represendationIndex = gst_mpdparser_get_rep_idx_with_max_bandwidth (representations, 300000); assert_equals_int (represendationIndex, 1); represendationIndex = gst_mpdparser_get_rep_idx_with_max_bandwidth (representations, 500000); assert_equals_int (represendationIndex, 0); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling Active stream selection * */ GST_START_TEST (dash_mpdparser_activeStream_selection) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; guint activeStreams; GstActiveStream *activeStream; const gchar *xml = "" "" " " " " " " " " " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* no active streams yet */ activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, 0); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); /* 1 active streams */ activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, 1); /* setup streaming from the second adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 1); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); /* 2 active streams */ activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, 2); /* setup streaming from the third adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 2); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); /* 3 active streams */ activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, 3); /* get details of the first active stream */ activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); assert_equals_int (activeStream->mimeType, GST_STREAM_VIDEO); /* get details of the second active stream */ activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 1); fail_if (activeStream == NULL); assert_equals_int (activeStream->mimeType, GST_STREAM_AUDIO); /* get details of the third active stream */ activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 2); fail_if (activeStream == NULL); assert_equals_int (activeStream->mimeType, GST_STREAM_APPLICATION); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test getting Active stream parameters * */ GST_START_TEST (dash_mpdparser_activeStream_parameters) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; guint activeStreams; GstActiveStream *activeStream; const gchar *mimeType; gboolean bitstreamSwitchingFlag; guint videoStreamWidth; guint videoStreamHeight; guint audioStreamRate; guint audioChannelsCount; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); /* 1 active streams */ activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, 1); /* get details of the first active stream */ activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); assert_equals_int (activeStream->mimeType, GST_STREAM_VIDEO); mimeType = gst_mpd_client_get_stream_mimeType (activeStream); assert_equals_string (mimeType, "video/quicktime"); bitstreamSwitchingFlag = gst_mpd_client_get_bitstream_switching_flag (activeStream); assert_equals_int (bitstreamSwitchingFlag, 1); videoStreamWidth = gst_mpd_client_get_video_stream_width (activeStream); assert_equals_int (videoStreamWidth, 320); videoStreamHeight = gst_mpd_client_get_video_stream_height (activeStream); assert_equals_int (videoStreamHeight, 240); audioStreamRate = gst_mpd_client_get_audio_stream_rate (activeStream); assert_equals_int (audioStreamRate, 48000); audioChannelsCount = gst_mpd_client_get_audio_stream_num_channels (activeStream); assert_equals_int (audioChannelsCount, 0); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test getting number and list of audio languages * */ GST_START_TEST (dash_mpdparser_get_audio_languages) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; guint activeStreams; guint adaptationSetsCount; GList *languages = NULL; guint languagesCount; const gchar *xml = "" "" " " " " " " " " " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from all adaptation sets */ adaptationSetsCount = gst_mpdparser_get_nb_adaptationSet (mpdclient); for (int i = 0; i < adaptationSetsCount; i++) { adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, i); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); } activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, adaptationSetsCount); languagesCount = gst_mpdparser_get_list_and_nb_of_audio_language (mpdclient, &languages); assert_equals_int (languagesCount, 2); assert_equals_string ((gchar *) g_list_nth_data (languages, 0), "en"); assert_equals_string ((gchar *) g_list_nth_data (languages, 1), "fr"); g_list_free (languages); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test getting the base URL * */ GST_START_TEST (dash_mpdparser_get_baseURL) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; guint activeStreams; guint adaptationSetsCount; const gchar *baseURL; const gchar *xml = "" "" " http://example.com/" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from all adaptation sets */ adaptationSetsCount = gst_mpdparser_get_nb_adaptationSet (mpdclient); for (int i = 0; i < adaptationSetsCount; i++) { adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, i); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); } activeStreams = gst_mpdparser_get_nb_active_stream (mpdclient); assert_equals_int (activeStreams, adaptationSetsCount); baseURL = gst_mpdparser_get_baseURL (mpdclient, 0); fail_if (baseURL == NULL); assert_equals_string (baseURL, "http://example.com/"); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test getting mediaPresentationDuration * */ GST_START_TEST (dash_mpdparser_get_mediaPresentationDuration) { GstClockTime mediaPresentationDuration; const gchar *xml = "" ""; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); mediaPresentationDuration = gst_mpd_client_get_media_presentation_duration (mpdclient); assert_equals_uint64 (mediaPresentationDuration, 3000000000); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test getting streamPresentationOffset * */ GST_START_TEST (dash_mpdparser_get_streamPresentationOffset) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; GstClockTime offset; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); /* test the stream presentation time offset */ offset = gst_mpd_parser_get_stream_presentation_offset (mpdclient, 0); /* seems to be set only for template segments, so here it is 0 */ assert_equals_int (offset, 0); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling segments * */ GST_START_TEST (dash_mpdparser_segments) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; gboolean hasNextSegment; GstActiveStream *activeStream; GstFlowReturn flow; GstMediaSegment segment; GstDateTime *segmentEndTime; GstDateTime *gst_time; GDateTime *g_time; GstClockTime ts; gint64 diff; const gchar *xml = "" "" " " " " " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); /* segment_index 0, segment_count 2. * Has next segment and can advance to next segment */ hasNextSegment = gst_mpd_client_has_next_segment (mpdclient, activeStream, TRUE); assert_equals_int (hasNextSegment, 1); flow = gst_mpd_client_advance_segment (mpdclient, activeStream, TRUE); assert_equals_int (flow, GST_FLOW_OK); /* segment_index 1, segment_count 2. * Does not have next segment and can not advance to next segment */ hasNextSegment = gst_mpd_client_has_next_segment (mpdclient, activeStream, TRUE); assert_equals_int (hasNextSegment, 0); flow = gst_mpd_client_advance_segment (mpdclient, activeStream, TRUE); assert_equals_int (flow, GST_FLOW_EOS); /* go to first segment */ gst_mpd_client_seek_to_first_segment (mpdclient); /* segment_index 0, segment_count 2. * Has next segment and can advance to next segment */ hasNextSegment = gst_mpd_client_has_next_segment (mpdclient, activeStream, TRUE); assert_equals_int (hasNextSegment, 1); flow = gst_mpd_client_advance_segment (mpdclient, activeStream, TRUE); assert_equals_int (flow, GST_FLOW_OK); /* segment_index 1, segment_count 2 * Does not have next segment */ hasNextSegment = gst_mpd_client_has_next_segment (mpdclient, activeStream, TRUE); assert_equals_int (hasNextSegment, 0); /* get chunk 0. segment_index will not change */ ret = gst_mpdparser_get_chunk_by_index (mpdclient, 0, 0, &segment); assert_equals_int (ret, 1); assert_equals_int (segment.number, 1); /* segment index is still 1 */ hasNextSegment = gst_mpd_client_has_next_segment (mpdclient, activeStream, TRUE); assert_equals_int (hasNextSegment, 0); /* each segment has a duration of 0 hours, 0 min 45 seconds * segment index is 1. * End time is at the end of segment 1, so 2 * segment_duration = 2 * 45s */ segmentEndTime = gst_mpd_client_get_next_segment_availability_end_time (mpdclient, activeStream); assert_equals_int (gst_date_time_get_year (segmentEndTime), 2015); assert_equals_int (gst_date_time_get_month (segmentEndTime), 3); assert_equals_int (gst_date_time_get_day (segmentEndTime), 24); assert_equals_int (gst_date_time_get_hour (segmentEndTime), 0); assert_equals_int (gst_date_time_get_minute (segmentEndTime), 1); assert_equals_int (gst_date_time_get_second (segmentEndTime), 30); gst_date_time_unref (segmentEndTime); /* seek to time */ gst_time = gst_date_time_new_from_iso8601_string ("2015-03-24T0:0:20Z"); g_time = gst_date_time_to_g_date_time (gst_time); ret = gst_mpd_client_seek_to_time (mpdclient, g_time); assert_equals_int (ret, 1); gst_date_time_unref (gst_time); g_date_time_unref (g_time); /* segment index is now 0 */ hasNextSegment = gst_mpd_client_has_next_segment (mpdclient, activeStream, TRUE); assert_equals_int (hasNextSegment, 1); /* check if stream at moment ts is available. * timeShiftBufferDepth was not set, so it is considered infinite. * All segments from the past must be available */ ts = 30 * GST_SECOND; ret = gst_mpd_client_check_time_position (mpdclient, activeStream, ts, &diff); assert_equals_int (ret, 0); assert_equals_int64 (diff, 0); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling headers * */ GST_START_TEST (dash_mpdparser_headers) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; gchar *uri; gint64 range_start; gint64 range_end; const gchar *xml = "" "" " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); /* get segment url and range from segment Initialization */ ret = gst_mpd_client_get_next_header (mpdclient, &uri, 0, &range_start, &range_end); assert_equals_int (ret, TRUE); assert_equals_string (uri, "TestSourceUrl"); assert_equals_int64 (range_start, 100); assert_equals_int64 (range_end, 200); g_free (uri); /* get segment url and range from segment indexRange */ ret = gst_mpd_client_get_next_header_index (mpdclient, &uri, 0, &range_start, &range_end); assert_equals_int (ret, TRUE); assert_equals_string (uri, "TestSourceUrl"); assert_equals_int64 (range_start, 10); assert_equals_int64 (range_end, 20); g_free (uri); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling fragments * */ GST_START_TEST (dash_mpdparser_fragments) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; GstMediaFragmentInfo fragment; GstActiveStream *activeStream; GstClockTime nextFragmentDuration; GstClockTime nextFragmentTimestamp; GstClockTime nextFragmentTimestampEnd; GstClockTime expectedDuration; GstClockTime expectedTimestamp; GstClockTime expectedTimestampEnd; const gchar *xml = "" "" " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); /* expected duration of the next fragment */ expectedDuration = duration_to_ms (0, 0, 0, 3, 3, 20, 0); expectedTimestamp = duration_to_ms (0, 0, 0, 0, 0, 10, 0); expectedTimestampEnd = duration_to_ms (0, 0, 0, 3, 3, 30, 0); ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, ""); assert_equals_int64 (fragment.range_start, 0); assert_equals_int64 (fragment.range_end, -1); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); nextFragmentDuration = gst_mpd_client_get_next_fragment_duration (mpdclient, activeStream); assert_equals_uint64 (nextFragmentDuration, expectedDuration * GST_MSECOND); ret = gst_mpd_client_get_next_fragment_timestamp (mpdclient, 0, &nextFragmentTimestamp); assert_equals_int (ret, TRUE); assert_equals_uint64 (nextFragmentTimestamp, expectedTimestamp * GST_MSECOND); ret = gst_mpd_client_get_last_fragment_timestamp_end (mpdclient, 0, &nextFragmentTimestampEnd); assert_equals_int (ret, TRUE); assert_equals_uint64 (nextFragmentTimestampEnd, expectedTimestampEnd * GST_MSECOND); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test inheriting segmentBase from parent * */ GST_START_TEST (dash_mpdparser_inherited_segmentBase) { GstPeriodNode *periodNode; GstSegmentBaseType *segmentBase; GstAdaptationSetNode *adaptationSet; GstRepresentationNode *representation; const gchar *xml = "" "" " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); periodNode = (GstPeriodNode *) mpdclient->mpd_node->Periods->data; adaptationSet = (GstAdaptationSetNode *) periodNode->AdaptationSets->data; representation = (GstRepresentationNode *) adaptationSet->Representations->data; /* test segment base from adaptation set */ segmentBase = adaptationSet->SegmentBase; assert_equals_uint64 (segmentBase->timescale, 100); /* test segment base from representation */ segmentBase = representation->SegmentBase; assert_equals_uint64 (segmentBase->timescale, 200); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test inheriting segmentURL from parent * */ GST_START_TEST (dash_mpdparser_inherited_segmentURL) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; GstActiveStream *activeStream; GstMediaFragmentInfo fragment; GstClockTime expectedDuration; GstClockTime expectedTimestamp; GstFlowReturn flow; const gchar *xml = "" "" " " " " " " " " " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); /* expected duration of the next fragment * Segment duration was set to 100 in AdaptationSet and to 110 in Representation * We expect duration to be 110 */ expectedDuration = duration_to_ms (0, 0, 0, 0, 0, 110, 0); expectedTimestamp = duration_to_ms (0, 0, 0, 0, 0, 10, 0); /* the representation contains 2 segments * - one inherited from AdaptationSet (duration 100) * - the second defined in the Representation (duration 110) * * Both will have the duration specified in the Representation (110) */ /* check first segment */ ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMediaAdaptation"); assert_equals_int64 (fragment.range_start, 10); assert_equals_int64 (fragment.range_end, 20); assert_equals_string (fragment.index_uri, "/TestIndexAdaptation"); assert_equals_int64 (fragment.index_range_start, 30); assert_equals_int64 (fragment.index_range_end, 40); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); /* advance to next segment */ flow = gst_mpd_client_advance_segment (mpdclient, activeStream, TRUE); assert_equals_int (flow, GST_FLOW_OK); /* second segment starts after first ends */ expectedTimestamp = expectedTimestamp + expectedDuration; /* check second segment */ ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMediaRep"); assert_equals_int64 (fragment.range_start, 100); assert_equals_int64 (fragment.range_end, 200); assert_equals_string (fragment.index_uri, "/TestIndexRep"); assert_equals_int64 (fragment.index_range_start, 300); assert_equals_int64 (fragment.index_range_end, 400); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test segment list * */ GST_START_TEST (dash_mpdparser_segment_list) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; GstActiveStream *activeStream; GstMediaFragmentInfo fragment; GstClockTime expectedDuration; GstClockTime expectedTimestamp; const gchar *xml = "" "" " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); /* expected duration of the next fragment * Segment duration was set larger than period duration (12000 vs 11000). * We expect it to be limited to period duration. */ expectedDuration = duration_to_ms (0, 0, 0, 3, 3, 20, 0); expectedTimestamp = duration_to_ms (0, 0, 0, 0, 0, 10, 0); ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMedia"); assert_equals_int64 (fragment.range_start, 100); assert_equals_int64 (fragment.range_end, 200); assert_equals_string (fragment.index_uri, "/TestIndex"); assert_equals_int64 (fragment.index_range_start, 300); assert_equals_int64 (fragment.index_range_end, 400); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test segment template * */ GST_START_TEST (dash_mpdparser_segment_template) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; GstActiveStream *activeStream; GstMediaFragmentInfo fragment; GstClockTime expectedDuration; GstClockTime expectedTimestamp; const gchar *xml = "" "" " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); /* expected duration of the next fragment * Segment duration was set larger than period duration (12000 vs 11000). * We expect it to not be limited to period duration. */ expectedDuration = duration_to_ms (0, 0, 0, 0, 0, 12000, 0); expectedTimestamp = duration_to_ms (0, 0, 0, 0, 0, 10, 0); ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMedia_rep=repIdnumber=1bandwidth=250000time=0"); assert_equals_int64 (fragment.range_start, 0); assert_equals_int64 (fragment.range_end, -1); assert_equals_string (fragment.index_uri, "/TestIndex"); assert_equals_int64 (fragment.index_range_start, 0); assert_equals_int64 (fragment.index_range_end, -1); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test segment timeline * */ GST_START_TEST (dash_mpdparser_segment_timeline) { GList *adaptationSets; GstAdaptationSetNode *adapt_set; GstActiveStream *activeStream; GstMediaFragmentInfo fragment; GstClockTime expectedDuration; GstClockTime expectedTimestamp; GstFlowReturn flow; const gchar *xml = "" "" " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* get the list of adaptation sets of the first period */ adaptationSets = gst_mpd_client_get_adaptation_sets (mpdclient); fail_if (adaptationSets == NULL); /* setup streaming from the first adaptation set */ adapt_set = (GstAdaptationSetNode *) g_list_nth_data (adaptationSets, 0); fail_if (adapt_set == NULL); ret = gst_mpd_client_setup_streaming (mpdclient, adapt_set); assert_equals_int (ret, TRUE); activeStream = gst_mpdparser_get_active_stream_by_index (mpdclient, 0); fail_if (activeStream == NULL); /* expected duration of the next fragment */ expectedDuration = duration_to_ms (0, 0, 0, 0, 0, 2, 0); expectedTimestamp = duration_to_ms (0, 0, 0, 0, 0, 13, 0); ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMedia0"); assert_equals_string (fragment.index_uri, "/TestIndex0"); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); /* advance to next segment */ flow = gst_mpd_client_advance_segment (mpdclient, activeStream, TRUE); assert_equals_int (flow, GST_FLOW_OK); /* second segment starts after first ends */ expectedTimestamp = expectedTimestamp + expectedDuration; /* check second segment. * It is a repeat of first segmentURL, because "r" in SegmentTimeline is 1 */ ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMedia0"); assert_equals_string (fragment.index_uri, "/TestIndex0"); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); /* advance to next segment */ flow = gst_mpd_client_advance_segment (mpdclient, activeStream, TRUE); assert_equals_int (flow, GST_FLOW_OK); /* third segment has a small gap after the second ends (t=10) * We also need to take into consideration period's start (10) */ expectedDuration = duration_to_ms (0, 0, 0, 0, 0, 3, 0); expectedTimestamp = duration_to_ms (0, 0, 0, 0, 0, 20, 0); /* check third segment */ ret = gst_mpd_client_get_next_fragment (mpdclient, 0, &fragment); assert_equals_int (ret, TRUE); assert_equals_string (fragment.uri, "/TestMedia1"); assert_equals_string (fragment.index_uri, "/TestIndex1"); assert_equals_uint64 (fragment.duration, expectedDuration * GST_MSECOND); assert_equals_uint64 (fragment.timestamp, expectedTimestamp * GST_MSECOND); gst_media_fragment_info_clear (&fragment); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing empty xml string * */ GST_START_TEST (dash_mpdparser_missing_xml) { const gchar *xml = ""; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, FALSE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing an xml with no mpd tag * */ GST_START_TEST (dash_mpdparser_missing_mpd) { const gchar *xml = ""; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, FALSE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing an MPD with a wrong end tag */ GST_START_TEST (dash_mpdparser_no_end_tag) { const gchar *xml = "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, FALSE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test parsing an MPD with no default namespace */ GST_START_TEST (dash_mpdparser_no_default_namespace) { const gchar *xml = "" ""; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, strlen (xml)); assert_equals_int (ret, TRUE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling wrong period duration during attempts to * infer a period duration from the start time of the next period */ GST_START_TEST (dash_mpdparser_wrong_period_duration_inferred_from_next_period) { const gchar *periodName; const gchar *xml = "" "" " " " " " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* period_idx should be 0 and we should have no active periods */ assert_equals_uint64 (mpdclient->period_idx, 0); fail_unless (mpdclient->periods == NULL); /* process the xml data */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, TRUE); /* Period0 should be present */ fail_unless (mpdclient->periods != NULL); periodName = gst_mpd_client_get_period_id (mpdclient); assert_equals_string (periodName, "Period0"); /* Period1 should not be present due to wrong duration */ ret = gst_mpd_client_set_period_index (mpdclient, 1); assert_equals_int (ret, FALSE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * Test handling wrong period duration during attempts to * infer a period duration from the mediaPresentationDuration */ GST_START_TEST (dash_mpdparser_wrong_period_duration_inferred_from_next_mediaPresentationDuration) { const gchar *xml = "" "" " "; gboolean ret; GstMpdClient *mpdclient = gst_mpd_client_new (); ret = gst_mpd_parse (mpdclient, xml, (gint) strlen (xml)); assert_equals_int (ret, TRUE); /* period_idx should be 0 and we should have no active periods */ assert_equals_uint64 (mpdclient->period_idx, 0); fail_unless (mpdclient->periods == NULL); /* process the xml data * should fail due to wrong duration in Period0 (start > mediaPresentationDuration) */ ret = gst_mpd_client_setup_media_presentation (mpdclient); assert_equals_int (ret, FALSE); gst_mpd_client_free (mpdclient); } GST_END_TEST; /* * create a test suite containing all dash testcases */ static Suite * dash_suite (void) { Suite *s = suite_create ("dash"); TCase *tc_simpleMPD = tcase_create ("simpleMPD"); TCase *tc_complexMPD = tcase_create ("complexMPD"); TCase *tc_negativeTests = tcase_create ("negativeTests"); GST_DEBUG_CATEGORY_INIT (gst_dash_demux_debug, "gst_dash_demux_debug", 0, "mpeg dash tests"); /* test parsing the simplest possible mpd */ tcase_add_test (tc_simpleMPD, dash_mpdparser_validsimplempd); /* tests parsing attributes from each element type */ tcase_add_test (tc_simpleMPD, dash_mpdparser_mpd); tcase_add_test (tc_simpleMPD, dash_mpdparser_programInformation); tcase_add_test (tc_simpleMPD, dash_mpdparser_baseURL); tcase_add_test (tc_simpleMPD, dash_mpdparser_location); tcase_add_test (tc_simpleMPD, dash_mpdparser_metrics); tcase_add_test (tc_simpleMPD, dash_mpdparser_metrics_range); tcase_add_test (tc_simpleMPD, dash_mpdparser_metrics_reporting); tcase_add_test (tc_simpleMPD, dash_mpdparser_period); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_baseURL); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentBase); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentBase_initialization); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentBase_representationIndex); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList_multipleSegmentBaseType); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList_multipleSegmentBaseType_segmentBaseType); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList_multipleSegmentBaseType_segmentTimeline); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList_multipleSegmentBaseType_segmentTimeline_s); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList_multipleSegmentBaseType_bitstreamSwitching); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentList_segmentURL); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentTemplate); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_segmentBaseType); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_segmentTimeline); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_segmentTimeline_s); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_segmentTemplate_multipleSegmentBaseType_bitstreamSwitching); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representationBase); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representationBase_framePacking); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representationBase_audioChannelConfiguration); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representationBase_contentProtection); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_accessibility); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_role); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_rating); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_viewpoint); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_contentComponent); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_contentComponent_accessibility); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_contentComponent_role); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_contentComponent_rating); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_contentComponent_viewpoint); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_baseURL); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_segmentBase); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_segmentBase_initialization); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_segmentBase_representationIndex); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_segmentList); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_segmentTemplate); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_representationBase); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_baseURL); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_subRepresentation); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_subRepresentation_representationBase); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_segmentBase); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_segmentList); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_adaptationSet_representation_segmentTemplate); tcase_add_test (tc_simpleMPD, dash_mpdparser_period_subset); /* tests checking other possible values for attributes */ tcase_add_test (tc_simpleMPD, dash_mpdparser_type_dynamic); tcase_add_test (tc_simpleMPD, dash_mpdparser_template_parsing); tcase_add_test (tc_simpleMPD, dash_mpdparser_isoff_ondemand_profile); tcase_add_test (tc_simpleMPD, dash_mpdparser_GstDateTime); /* tests checking the MPD management * (eg. setting active streams, obtaining attributes values) */ tcase_add_test (tc_complexMPD, dash_mpdparser_setup_media_presentation); tcase_add_test (tc_complexMPD, dash_mpdparser_setup_streaming); tcase_add_test (tc_complexMPD, dash_mpdparser_period_selection); tcase_add_test (tc_complexMPD, dash_mpdparser_get_period_at_time); tcase_add_test (tc_complexMPD, dash_mpdparser_adaptationSet_handling); tcase_add_test (tc_complexMPD, dash_mpdparser_representation_selection); tcase_add_test (tc_complexMPD, dash_mpdparser_activeStream_selection); tcase_add_test (tc_complexMPD, dash_mpdparser_activeStream_parameters); tcase_add_test (tc_complexMPD, dash_mpdparser_get_audio_languages); tcase_add_test (tc_complexMPD, dash_mpdparser_get_baseURL); tcase_add_test (tc_complexMPD, dash_mpdparser_get_mediaPresentationDuration); tcase_add_test (tc_complexMPD, dash_mpdparser_get_streamPresentationOffset); tcase_add_test (tc_complexMPD, dash_mpdparser_segments); tcase_add_test (tc_complexMPD, dash_mpdparser_headers); tcase_add_test (tc_complexMPD, dash_mpdparser_fragments); tcase_add_test (tc_complexMPD, dash_mpdparser_inherited_segmentBase); tcase_add_test (tc_complexMPD, dash_mpdparser_inherited_segmentURL); tcase_add_test (tc_complexMPD, dash_mpdparser_segment_list); tcase_add_test (tc_complexMPD, dash_mpdparser_segment_template); tcase_add_test (tc_complexMPD, dash_mpdparser_segment_timeline); /* tests checking the parsing of missing/incomplete attributes of xml */ tcase_add_test (tc_negativeTests, dash_mpdparser_missing_xml); tcase_add_test (tc_negativeTests, dash_mpdparser_missing_mpd); tcase_add_test (tc_negativeTests, dash_mpdparser_no_end_tag); tcase_add_test (tc_negativeTests, dash_mpdparser_no_default_namespace); tcase_add_test (tc_negativeTests, dash_mpdparser_wrong_period_duration_inferred_from_next_period); tcase_add_test (tc_negativeTests, dash_mpdparser_wrong_period_duration_inferred_from_next_mediaPresentationDuration); suite_add_tcase (s, tc_simpleMPD); suite_add_tcase (s, tc_complexMPD); suite_add_tcase (s, tc_negativeTests); return s; } GST_CHECK_MAIN (dash);