/* GStreamer Editing Services * * Copyright (C) <2013> Thibault Saunier * * 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., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "ges-validate.h" #include #ifdef HAVE_GST_VALIDATE #include #include #include #define MONITOR_ON_PIPELINE "validate-monitor" #define RUNNER_ON_PIPELINE "runner-monitor" static void _validate_report_added_cb (GstValidateRunner * runner, GstValidateReport * report, GstPipeline * pipeline) { if (report->level == GST_VALIDATE_REPORT_LEVEL_CRITICAL) { GST_DEBUG_BIN_TO_DOT_FILE_WITH_TS (GST_BIN (pipeline), GST_DEBUG_GRAPH_SHOW_ALL, "ges-launch--validate-error"); } } static GESTimeline * get_timeline (GstValidateScenario * scenario) { GESTimeline *timeline; g_object_get (scenario->pipeline, "timeline", &timeline, NULL); return timeline; } static gboolean _set_child_property (GstValidateScenario * scenario, GstValidateAction * action) { const GValue *value; GESTimeline *timeline; GESTimelineElement *element; const gchar *property_name, *element_name; element_name = gst_structure_get_string (action->structure, "element-name"); timeline = get_timeline (scenario); g_return_val_if_fail (timeline, FALSE); element = ges_timeline_get_element (timeline, element_name); g_return_val_if_fail (GES_IS_TRACK_ELEMENT (element), FALSE); property_name = gst_structure_get_string (action->structure, "property"); value = gst_structure_get_value (action->structure, "value"); GST_DEBUG ("%s Setting %s property to %p", element->name, property_name, value); ges_track_element_set_child_property (GES_TRACK_ELEMENT (element), property_name, (GValue *) value); g_object_unref (timeline); return TRUE; } static gboolean _serialize_project (GstValidateScenario * scenario, GstValidateAction * action) { const gchar *uri = gst_structure_get_string (action->structure, "uri"); GESTimeline *timeline = get_timeline (scenario); gboolean res; gst_validate_printf (action, "Saving project to %s", uri); res = ges_timeline_save_to_uri (timeline, uri, NULL, TRUE, NULL); g_object_unref (timeline); return res; } static gboolean _remove_asset (GstValidateScenario * scenario, GstValidateAction * action) { const gchar *id = NULL; const gchar *type_string = NULL; GType type; GESTimeline *timeline = get_timeline (scenario); GESProject *project = ges_timeline_get_project (timeline); GESAsset *asset; gboolean res = FALSE; id = gst_structure_get_string (action->structure, "id"); type_string = gst_structure_get_string (action->structure, "type"); if (!type_string || !id) { GST_ERROR ("Missing parameters, we got type %s and id %s", type_string, id); goto beach; } if (!(type = g_type_from_name (type_string))) { GST_ERROR ("This type doesn't exist : %s", type_string); goto beach; } asset = ges_project_get_asset (project, id, type); if (!asset) { GST_ERROR ("No asset with id %s and type %s", id, type_string); goto beach; } res = ges_project_remove_asset (project, asset); beach: g_object_unref (timeline); return res; } static gboolean _add_asset (GstValidateScenario * scenario, GstValidateAction * action) { const gchar *id = NULL; const gchar *type_string = NULL; GType type; GESTimeline *timeline = get_timeline (scenario); GESProject *project = ges_timeline_get_project (timeline); GESAsset *asset; GError *error = NULL; gboolean res = FALSE; id = gst_structure_get_string (action->structure, "id"); type_string = gst_structure_get_string (action->structure, "type"); if (!type_string || !id) { GST_ERROR ("Missing parameters, we got type %s and id %s", type_string, id); goto beach; } if (!(type = g_type_from_name (type_string))) { GST_ERROR ("This type doesn't exist : %s", type_string); goto beach; } if (type == GES_TYPE_URI_CLIP) asset = (GESAsset *) ges_uri_clip_asset_request_sync (id, &error); else asset = ges_asset_request (type, id, &error); if (!asset || error) { GST_ERROR ("There was an error requesting the asset with id %s and type %s", id, type_string); goto beach; } res = ges_project_add_asset (project, asset); beach: g_object_unref (timeline); return res; } /* Unref after usage */ static GESLayer * _get_layer_by_priority (GESTimeline * timeline, gint priority) { GList *layers, *tmp; GESLayer *layer = NULL; layers = ges_timeline_get_layers (timeline); for (tmp = layers; tmp; tmp = tmp->next) { GESLayer *tmp_layer = GES_LAYER (tmp->data); guint tmp_priority; g_object_get (tmp_layer, "priority", &tmp_priority, NULL); if ((gint) tmp_priority == priority) { layer = gst_object_ref (tmp_layer); break; } } g_list_free_full (layers, gst_object_unref); return layer; } static gboolean _add_layer (GstValidateScenario * scenario, GstValidateAction * action) { GESTimeline *timeline = get_timeline (scenario); GESLayer *layer; gint priority; gboolean res = FALSE, auto_transition = FALSE; if (!gst_structure_get_int (action->structure, "priority", &priority)) { GST_ERROR ("priority is needed when adding a layer"); goto beach; } layer = _get_layer_by_priority (timeline, priority); if (layer != NULL) { GST_ERROR ("A layer with priority %d already exists, not creating a new one", priority); gst_object_unref (layer); goto beach; } gst_structure_get_boolean (action->structure, "auto-transition", &auto_transition); layer = ges_layer_new (); g_object_set (layer, "priority", priority, "auto-transition", auto_transition, NULL); res = ges_timeline_add_layer (timeline, layer); beach: g_object_unref (timeline); return res; } static gboolean _remove_layer (GstValidateScenario * scenario, GstValidateAction * action) { GESTimeline *timeline = get_timeline (scenario); GESLayer *layer; gint priority; gboolean res = FALSE; if (!gst_structure_get_int (action->structure, "priority", &priority)) { GST_ERROR ("priority is needed when removing a layer"); goto beach; } layer = _get_layer_by_priority (timeline, priority); if (layer) { res = ges_timeline_remove_layer (timeline, layer); gst_object_unref (layer); } else { GST_ERROR ("No layer with priority %d", priority); } beach: g_object_unref (timeline); return res; } static gboolean _remove_clip (GstValidateScenario * scenario, GstValidateAction * action) { GESTimeline *timeline = get_timeline (scenario); GESTimelineElement *clip; GESLayer *layer; const gchar *name; gboolean res = FALSE; name = gst_structure_get_string (action->structure, "name"); clip = ges_timeline_get_element (timeline, name); g_return_val_if_fail (GES_IS_CLIP (clip), FALSE); layer = ges_clip_get_layer (GES_CLIP (clip)); if (layer) { res = ges_layer_remove_clip (layer, GES_CLIP (clip)); gst_object_unref (layer); } else { GST_ERROR ("No layer for clip %s", ges_timeline_element_get_name (clip)); } g_object_unref (timeline); return res; } static gboolean _add_clip (GstValidateScenario * scenario, GstValidateAction * action) { GESTimeline *timeline = get_timeline (scenario); GESAsset *asset; GESLayer *layer; GESClip *clip; GError *error = NULL; gint layer_priority; const gchar *name; const gchar *asset_id; const gchar *type_string; GType type; gboolean res = FALSE; GstClockTime duration = 1 * GST_SECOND, inpoint = 0, start = GST_CLOCK_TIME_NONE; gst_structure_get_int (action->structure, "layer-priority", &layer_priority); name = gst_structure_get_string (action->structure, "name"); asset_id = gst_structure_get_string (action->structure, "asset-id"); type_string = gst_structure_get_string (action->structure, "type"); gst_validate_action_get_clocktime (scenario, action, "start", &start); gst_validate_action_get_clocktime (scenario, action, "inpoint", &inpoint); gst_validate_action_get_clocktime (scenario, action, "duration", &duration); if (!(type = g_type_from_name (type_string))) { GST_ERROR ("This type doesn't exist : %s", type_string); goto beach; } asset = ges_asset_request (type, asset_id, &error); if (!asset || error) { GST_ERROR ("There was an error requesting the asset with id %s and type %s (%s)", asset_id, type_string, error->message); goto beach; } layer = _get_layer_by_priority (timeline, layer_priority); if (!layer) { GST_ERROR ("No layer with priority %d", layer_priority); goto beach; } if (type == GES_TYPE_URI_CLIP) { duration = GST_CLOCK_TIME_NONE; } clip = ges_layer_add_asset (layer, asset, start, inpoint, duration, GES_TRACK_TYPE_UNKNOWN); if (clip) { res = TRUE; if (!ges_timeline_element_set_name (GES_TIMELINE_ELEMENT (clip), name)) { res = FALSE; GST_ERROR ("couldn't set name %s on clip with id %s", name, asset_id); } } else { GST_ERROR ("Couldn't add clip with id %s to layer with priority %d", asset_id, layer_priority); } gst_object_unref (layer); beach: g_object_unref (timeline); return res; } static gboolean _edit_container (GstValidateScenario * scenario, GstValidateAction * action) { gint64 cpos; GList *layers = NULL; GESTimeline *timeline; GstQuery *query_segment; GESTimelineElement *container; GstClockTime position; gboolean res = FALSE; gint new_layer_priority = -1; GESEditMode edge = GES_EDGE_NONE; GESEditMode mode = GES_EDIT_MODE_NORMAL; const gchar *edit_mode_str = NULL, *edge_str = NULL; const gchar *clip_name; clip_name = gst_structure_get_string (action->structure, "container-name"); timeline = get_timeline (scenario); g_return_val_if_fail (timeline, FALSE); container = ges_timeline_get_element (timeline, clip_name); g_return_val_if_fail (GES_IS_CONTAINER (container), FALSE); if (!gst_validate_action_get_clocktime (scenario, action, "position", &position)) { GST_WARNING ("Could not get position"); goto beach; } if ((edit_mode_str = gst_structure_get_string (action->structure, "edit-mode"))) g_return_val_if_fail (gst_validate_utils_enum_from_str (GES_TYPE_EDIT_MODE, edit_mode_str, &mode), FALSE); if ((edge_str = gst_structure_get_string (action->structure, "edge"))) g_return_val_if_fail (gst_validate_utils_enum_from_str (GES_TYPE_EDGE, edge_str, &edge), FALSE); gst_structure_get_int (action->structure, "new-layer-priority", &new_layer_priority); gst_validate_printf (action, "Editing %s to %" GST_TIME_FORMAT " in %s mode, edge: %s " "with new layer prio: %d \n\n", clip_name, GST_TIME_ARGS (position), edit_mode_str ? edit_mode_str : "normal", edge_str ? edge_str : "None", new_layer_priority); if (!ges_container_edit (GES_CONTAINER (container), layers, new_layer_priority, mode, edge, position)) { gst_object_unref (container); goto beach; } gst_object_unref (container); query_segment = gst_query_new_segment (GST_FORMAT_TIME); if (!gst_element_query (scenario->pipeline, query_segment)) { GST_ERROR_OBJECT (scenario, "Could not query segment"); goto beach; } if (!gst_element_query_position (scenario->pipeline, GST_FORMAT_TIME, &cpos)) { GST_ERROR_OBJECT (scenario, "Could not query position"); goto beach; } beach: g_object_unref (timeline); return res; } static gboolean _commit (GstValidateScenario * scenario, GstValidateAction * action) { GESTimeline *timeline = get_timeline (scenario); gst_validate_printf (action, "Commiting timeline %s\n", GST_OBJECT_NAME (timeline)); ges_timeline_commit (timeline); gst_object_unref (timeline); return TRUE; } static gboolean _split_clip (GstValidateScenario * scenario, GstValidateAction * action) { GESTimeline *timeline; const gchar *clip_name; GESTimelineElement *element; GstClockTime position; clip_name = gst_structure_get_string (action->structure, "clip-name"); timeline = get_timeline (scenario); g_return_val_if_fail (timeline, FALSE); element = ges_timeline_get_element (timeline, clip_name); g_return_val_if_fail (GES_IS_CLIP (element), FALSE); g_object_unref (timeline); g_return_val_if_fail (gst_validate_action_get_clocktime (scenario, action, "position", &position), FALSE); return (ges_clip_split (GES_CLIP (element), position) != NULL); } static void ges_validate_register_action_types (void) { gst_validate_init (); /* *INDENT-OFF* */ gst_validate_register_action_type ("edit-container", "ges", _edit_container, (GstValidateActionParameter []) { { .name = "container-name", .description = "The name of the GESContainer to edit", .mandatory = TRUE, .types = "string", }, { .name = "position", .description = "The new position of the GESContainer", .mandatory = TRUE, .types = "double or string", .possible_variables = "position: The current position in the stream\n" "duration: The duration of the stream", NULL }, { .name = "edit-mode", .description = "The GESEditMode to use to edit @container-name", .mandatory = FALSE, .types = "string", .def = "normal", }, { .name = "edge", .description = "The GESEdge to use to edit @container-name\n" "should be in [ edge_start, edge_end, edge_none ] ", .mandatory = FALSE, .types = "string", .def = "edge_none", }, { .name = "new-layer-priority", .description = "The priority of the layer @container should land in.\n" "If the layer you're trying to move the container to doesn't exist, it will\n" "be created automatically. -1 means no move.", .mandatory = FALSE, .types = "int", .def = "-1", }, {NULL} }, "Allows to edit a container (like a GESClip), for more details, have a look at:\n" "ges_container_edit documentation, Note that the timeline will\n" "be commited, and flushed so that the edition is taken into account", FALSE); gst_validate_register_action_type ("add-asset", "ges", _add_asset, (GstValidateActionParameter []) { { .name = "id", .description = "", .mandatory = TRUE, NULL }, { .name = "type", .description = "The type of asset to add", .mandatory = TRUE, NULL }, {NULL} }, "Allows to add an asset to the current project", FALSE); gst_validate_register_action_type ("remove-asset", "ges", _remove_asset, (GstValidateActionParameter []) { { .name = "id", .description = "The ID of the clip to remove", .mandatory = TRUE, NULL }, { .name = "type", .description = "The type of asset to remove", .mandatory = TRUE, NULL }, { NULL } }, "Allows to remove an asset from the current project", FALSE); gst_validate_register_action_type ("add-layer", "ges", _add_layer, (GstValidateActionParameter []) { { .name = "priority", .description = "The priority of the new layer to add", .mandatory = TRUE, NULL }, { NULL } }, "Allows to add a layer to the current timeline", FALSE); gst_validate_register_action_type ("remove-layer", "ges", _remove_layer, (GstValidateActionParameter []) { { .name = "priority", .description = "The priority of the layer to remove", .mandatory = TRUE, NULL }, { .name = "auto-transition", .description = "Wheter auto-transition is activated on the new layer.", .mandatory = FALSE, .types="boolean", .def = "False" }, { NULL } }, "Allows to remove a layer from the current timeline", FALSE); gst_validate_register_action_type ("add-clip", "ges", _add_clip, (GstValidateActionParameter []) { { .name = "name", .description = "The name of the clip to add", .types = "string", .mandatory = TRUE, }, { .name = "layer-priority", .description = "The priority of the clip to add", .types = "int", .mandatory = TRUE, }, { .name = "asset-id", .description = "The id of the asset from which to extract the clip", .types = "string", .mandatory = TRUE, }, { .name = "type", .description = "The type of the clip to create", .types = "string", .mandatory = TRUE, }, { .name = "start", .description = "The start value to set on the new GESClip.", .types = "double or string", .mandatory = FALSE, }, { .name = "inpoint", .description = "The inpoint value to set on the new GESClip", .types = "double or string", .mandatory = FALSE, }, { .name = "duration", .description = "The duration value to set on the new GESClip", .types = "double or string", .mandatory = FALSE, }, {NULL} }, "Allows to add a clip to a given layer", FALSE); gst_validate_register_action_type ("remove-clip", "ges", _remove_clip, (GstValidateActionParameter []) { { .name = "name", .description = "The name of the clip to remove", .types = "string", .mandatory = TRUE, }, {NULL} }, "Allows to remove a clip from a given layer", FALSE); gst_validate_register_action_type ("serialize-project", "ges", _serialize_project, (GstValidateActionParameter []) { { .name = "uri", .description = "The uri where to store the serialized project", .types = "string", .mandatory = TRUE, }, {NULL} }, "serializes a project", FALSE); gst_validate_register_action_type ("set-child-property", "ges", _set_child_property, (GstValidateActionParameter []) { { .name = "element-name", .description = "The name of the element on which to modify the property", .types = "string", .mandatory = TRUE, }, { .name = "property", .description = "The name of the property to modify", .types = "string", .mandatory = TRUE, }, { .name = "value", .description = "The value of the property", .types = "gvalue", .mandatory = TRUE, }, {NULL} }, "Allows to change child property of an object", FALSE); gst_validate_register_action_type ("split-clip", "ges", _split_clip, (GstValidateActionParameter []) { { .name = "clip-name", .description = "The name of the clip to split", .types = "string", .mandatory = TRUE, }, { .name = "position", .description = "The position at which to split the clip", .types = "double or string", .mandatory = TRUE, }, {NULL} }, "Split a clip at a specified position.", FALSE); gst_validate_register_action_type ("commit", "ges", _commit, NULL, "Commit the timeline.", FALSE); /* *INDENT-ON* */ } gboolean ges_validate_activate (GstPipeline * pipeline, const gchar * scenario, gboolean * needs_setting_state) { GstValidateOverride *o; GstValidateRunner *runner = NULL; GstValidateMonitor *monitor = NULL; ges_validate_register_action_types (); if (scenario) { if (g_strcmp0 (scenario, "none")) { gchar *scenario_name = g_strconcat (scenario, "->gespipeline*", NULL); g_setenv ("GST_VALIDATE_SCENARIO", scenario_name, TRUE); g_free (scenario_name); } } o = gst_validate_override_new (); gst_validate_override_change_severity (o, EVENT_SEEK_RESULT_POSITION_WRONG, GST_VALIDATE_REPORT_LEVEL_WARNING); gst_validate_override_register_by_name ("scenarios", o); runner = gst_validate_runner_new (); g_signal_connect (runner, "report-added", G_CALLBACK (_validate_report_added_cb), pipeline); monitor = gst_validate_monitor_factory_create (GST_OBJECT_CAST (pipeline), runner, NULL); gst_validate_reporter_set_handle_g_logs (GST_VALIDATE_REPORTER (monitor)); g_object_get (monitor, "handles-states", needs_setting_state, NULL); *needs_setting_state = !*needs_setting_state; g_object_set_data (G_OBJECT (pipeline), MONITOR_ON_PIPELINE, monitor); g_object_set_data (G_OBJECT (pipeline), RUNNER_ON_PIPELINE, runner); return TRUE; } gint ges_validate_clean (GstPipeline * pipeline) { gint res = 0; GstValidateMonitor *monitor = g_object_get_data (G_OBJECT (pipeline), MONITOR_ON_PIPELINE); GstValidateRunner *runner = g_object_get_data (G_OBJECT (pipeline), RUNNER_ON_PIPELINE); if (runner) res = gst_validate_runner_printf (runner); gst_object_unref (pipeline); if (runner) { gst_object_unref (runner); if (monitor) gst_object_unref (monitor); } return res; } void ges_validate_handle_request_state_change (GstMessage * message, GMainLoop * mainloop) { GstState state; gst_message_parse_request_state (message, &state); if (GST_IS_VALIDATE_SCENARIO (GST_MESSAGE_SRC (message)) && state == GST_STATE_NULL) { gst_validate_printf (GST_MESSAGE_SRC (message), "State change request NULL, " "quiting mainloop\n"); g_main_loop_quit (mainloop); } } gint ges_validate_print_action_types (const gchar ** types, gint num_types) { ges_validate_register_action_types (); if (!gst_validate_print_action_types (types, num_types)) { GST_ERROR ("Could not print all wanted types"); return 1; } return 0; } #else static gboolean _print_position (GstElement * pipeline) { gint64 position, duration; if (pipeline) { gst_element_query_position (GST_ELEMENT (pipeline), GST_FORMAT_TIME, &position); gst_element_query_duration (GST_ELEMENT (pipeline), GST_FORMAT_TIME, &duration); g_print ("\r", GST_TIME_ARGS (position), GST_TIME_ARGS (duration)); } return TRUE; } gboolean ges_validate_activate (GstPipeline * pipeline, const gchar * scenario, gboolean * needs_setting_state) { if (scenario) { GST_WARNING ("Trying to run scenario %s, but gst-validate not supported", scenario); return FALSE; } g_object_set_data (G_OBJECT (pipeline), "pposition-id", GUINT_TO_POINTER (g_timeout_add (200, (GSourceFunc) _print_position, pipeline))); *needs_setting_state = TRUE; return TRUE; } gint ges_validate_clean (GstPipeline * pipeline) { g_source_remove (GPOINTER_TO_INT (g_object_get_data (G_OBJECT (pipeline), "pposition-id"))); gst_object_unref (pipeline); return 0; } void ges_validate_handle_request_state_change (GstMessage * message, GMainLoop * mainloop) { return; } gint ges_validate_print_action_types (const gchar ** types, gint num_types) { return 0; } #endif