asterisk/res/ari/resource_recordings.c

360 lines
9.6 KiB
C

/*
* Asterisk -- An open source telephony toolkit.
*
* Copyright (C) 2012 - 2013, Digium, Inc.
*
* David M. Lee, II <dlee@digium.com>
*
* See http://www.asterisk.org for more information about
* the Asterisk project. Please do not directly contact
* any of the maintainers of this project for assistance;
* the project provides a web site, mailing lists and IRC
* channels for your use.
*
* This program is free software, distributed under the terms of
* the GNU General Public License Version 2. See the LICENSE file
* at the top of the source tree.
*/
/*! \file
*
* \brief /api-docs/recordings.{format} implementation- Recording resources
*
* \author David M. Lee, II <dlee@digium.com>
*/
/*** MODULEINFO
<support_level>core</support_level>
***/
#include "asterisk.h"
#include "asterisk/stasis_app_recording.h"
#include "resource_recordings.h"
void ast_ari_recordings_list_stored(struct ast_variable *headers,
struct ast_ari_recordings_list_stored_args *args,
struct ast_ari_response *response)
{
RAII_VAR(struct ao2_container *, recordings, NULL, ao2_cleanup);
RAII_VAR(struct ast_json *, json, NULL, ast_json_unref);
struct ao2_iterator i;
void *obj;
recordings = stasis_app_stored_recording_find_all();
if (!recordings) {
ast_ari_response_alloc_failed(response);
return;
}
json = ast_json_array_create();
if (!json) {
ast_ari_response_alloc_failed(response);
return;
}
i = ao2_iterator_init(recordings, 0);
while ((obj = ao2_iterator_next(&i))) {
RAII_VAR(struct stasis_app_stored_recording *, recording, obj,
ao2_cleanup);
int r = ast_json_array_append(
json, stasis_app_stored_recording_to_json(recording));
if (r != 0) {
ast_ari_response_alloc_failed(response);
ao2_iterator_destroy(&i);
return;
}
}
ao2_iterator_destroy(&i);
ast_ari_response_ok(response, ast_json_ref(json));
}
void ast_ari_recordings_get_stored(struct ast_variable *headers,
struct ast_ari_recordings_get_stored_args *args,
struct ast_ari_response *response)
{
RAII_VAR(struct stasis_app_stored_recording *, recording, NULL,
ao2_cleanup);
struct ast_json *json;
recording = stasis_app_stored_recording_find_by_name(
args->recording_name);
if (recording == NULL) {
ast_ari_response_error(response, 404, "Not Found",
"Recording not found");
return;
}
json = stasis_app_stored_recording_to_json(recording);
if (json == NULL) {
ast_ari_response_error(response, 500,
"Internal Server Error", "Error building response");
return;
}
ast_ari_response_ok(response, json);
}
void ast_ari_recordings_get_stored_file(struct ast_tcptls_session_instance *ser,
struct ast_variable *headers, struct ast_ari_recordings_get_stored_file_args *args,
struct ast_ari_response *response)
{
RAII_VAR(struct stasis_app_stored_recording *, recording,
stasis_app_stored_recording_find_by_name(args->recording_name),
ao2_cleanup);
static const char *format_type_names[AST_MEDIA_TYPE_TEXT + 1] = {
[AST_MEDIA_TYPE_UNKNOWN] = "binary",
[AST_MEDIA_TYPE_AUDIO] = "audio",
[AST_MEDIA_TYPE_VIDEO] = "video",
[AST_MEDIA_TYPE_IMAGE] = "image",
[AST_MEDIA_TYPE_TEXT] = "text",
};
struct ast_format *format;
response->message = ast_json_null();
if (!recording) {
ast_ari_response_error(response, 404, "Not Found",
"Recording not found");
return;
}
format = ast_get_format_for_file_ext(stasis_app_stored_recording_get_extension(recording));
if (!format) {
ast_ari_response_error(response, 500, "Internal Server Error",
"Format specified by recording not available or loaded");
return;
}
response->fd = open(stasis_app_stored_recording_get_filename(recording), O_RDONLY);
if (response->fd < 0) {
ast_ari_response_error(response, 403, "Forbidden",
"Recording could not be opened");
return;
}
ast_str_append(&response->headers, 0, "Content-Type: %s/%s\r\n",
format_type_names[ast_format_get_type(format)],
stasis_app_stored_recording_get_extension(recording));
ast_ari_response_ok(response, ast_json_null());
}
void ast_ari_recordings_copy_stored(struct ast_variable *headers,
struct ast_ari_recordings_copy_stored_args *args,
struct ast_ari_response *response)
{
RAII_VAR(struct stasis_app_stored_recording *, src_recording, NULL,
ao2_cleanup);
RAII_VAR(struct stasis_app_stored_recording *, dst_recording, NULL,
ao2_cleanup);
struct ast_json *json;
int res;
src_recording = stasis_app_stored_recording_find_by_name(
args->recording_name);
if (src_recording == NULL) {
ast_ari_response_error(response, 404, "Not Found",
"Recording not found");
return;
}
dst_recording = stasis_app_stored_recording_find_by_name(
args->destination_recording_name);
if (dst_recording) {
ast_ari_response_error(response, 409, "Conflict",
"A recording with the same name already exists on the system");
return;
}
/* See if we got our name rejected */
switch (errno) {
case EINVAL:
ast_ari_response_error(response, 400, "Bad request",
"Invalid destination recording name");
return;
case EACCES:
ast_ari_response_error(response, 403, "Forbidden",
"Destination file path is forbidden");
return;
default:
break;
}
res = stasis_app_stored_recording_copy(src_recording,
args->destination_recording_name, &dst_recording);
if (res) {
switch (errno) {
case EACCES:
case EPERM:
ast_ari_response_error(response, 500,
"Internal Server Error",
"Copy failed");
break;
default:
ast_log(LOG_WARNING,
"Unexpected error copying recording %s to %s: %s\n",
args->recording_name, args->destination_recording_name, strerror(errno));
ast_ari_response_error(response, 500,
"Internal Server Error",
"Copy failed");
break;
}
return;
}
json = stasis_app_stored_recording_to_json(dst_recording);
if (json == NULL) {
ast_ari_response_error(response, 500,
"Internal Server Error", "Error building response");
return;
}
ast_ari_response_ok(response, json);
}
void ast_ari_recordings_delete_stored(struct ast_variable *headers,
struct ast_ari_recordings_delete_stored_args *args,
struct ast_ari_response *response)
{
RAII_VAR(struct stasis_app_stored_recording *, recording, NULL,
ao2_cleanup);
int res;
recording = stasis_app_stored_recording_find_by_name(
args->recording_name);
if (recording == NULL) {
ast_ari_response_error(response, 404, "Not Found",
"Recording not found");
return;
}
res = stasis_app_stored_recording_delete(recording);
if (res != 0) {
switch (errno) {
case EACCES:
case EPERM:
ast_ari_response_error(response, 500,
"Internal Server Error",
"Delete failed");
break;
default:
ast_log(LOG_WARNING,
"Unexpected error deleting recording %s: %s\n",
args->recording_name, strerror(errno));
ast_ari_response_error(response, 500,
"Internal Server Error",
"Delete failed");
break;
}
return;
}
ast_ari_response_no_content(response);
}
void ast_ari_recordings_get_live(struct ast_variable *headers,
struct ast_ari_recordings_get_live_args *args,
struct ast_ari_response *response)
{
RAII_VAR(struct stasis_app_recording *, recording, NULL, ao2_cleanup);
struct ast_json *json;
recording = stasis_app_recording_find_by_name(args->recording_name);
if (recording == NULL) {
ast_ari_response_error(response, 404, "Not Found",
"Recording not found");
return;
}
json = stasis_app_recording_to_json(recording);
if (json == NULL) {
ast_ari_response_error(response, 500,
"Internal Server Error", "Error building response");
return;
}
ast_ari_response_ok(response, json);
}
static void control_recording(const char *name,
enum stasis_app_recording_media_operation operation,
struct ast_ari_response *response)
{
RAII_VAR(struct stasis_app_recording *, recording, NULL, ao2_cleanup);
enum stasis_app_recording_oper_results res;
recording = stasis_app_recording_find_by_name(name);
if (recording == NULL) {
ast_ari_response_error(response, 404, "Not Found",
"Recording not found");
return;
}
res = stasis_app_recording_operation(recording, operation);
switch (res) {
case STASIS_APP_RECORDING_OPER_OK:
ast_ari_response_no_content(response);
return;
case STASIS_APP_RECORDING_OPER_FAILED:
ast_ari_response_error(response, 500,
"Internal Server Error", "Recording operation failed");
return;
case STASIS_APP_RECORDING_OPER_NOT_RECORDING:
ast_ari_response_error(response, 409,
"Conflict", "Recording not in session");
}
}
void ast_ari_recordings_cancel(struct ast_variable *headers,
struct ast_ari_recordings_cancel_args *args,
struct ast_ari_response *response)
{
control_recording(args->recording_name, STASIS_APP_RECORDING_CANCEL,
response);
}
void ast_ari_recordings_stop(struct ast_variable *headers,
struct ast_ari_recordings_stop_args *args,
struct ast_ari_response *response)
{
control_recording(args->recording_name, STASIS_APP_RECORDING_STOP,
response);
}
void ast_ari_recordings_pause(struct ast_variable *headers,
struct ast_ari_recordings_pause_args *args,
struct ast_ari_response *response)
{
control_recording(args->recording_name, STASIS_APP_RECORDING_PAUSE,
response);
}
void ast_ari_recordings_unpause(struct ast_variable *headers,
struct ast_ari_recordings_unpause_args *args,
struct ast_ari_response *response)
{
control_recording(args->recording_name, STASIS_APP_RECORDING_UNPAUSE,
response);
}
void ast_ari_recordings_mute(struct ast_variable *headers,
struct ast_ari_recordings_mute_args *args,
struct ast_ari_response *response)
{
control_recording(args->recording_name, STASIS_APP_RECORDING_MUTE,
response);
}
void ast_ari_recordings_unmute(struct ast_variable *headers,
struct ast_ari_recordings_unmute_args *args,
struct ast_ari_response *response)
{
control_recording(args->recording_name, STASIS_APP_RECORDING_UNMUTE,
response);
}