You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
sems/apps/ivr/IvrAudio.cpp

333 lines
8.3 KiB

#include "IvrAudio.h"
#include "AmAudio.h"
#include "AmSession.h"
#include "log.h"
#ifdef IVR_WITH_TTS
#define TTS_CACHE_PATH "/tmp/"
extern "C" cst_voice *register_cmu_us_kal();
#endif //ivr_with_tts
static PyObject* IvrAudioFile_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
{
DBG("---------- IvrAudioFile_alloc -----------\n");
IvrAudioFile *self;
self = (IvrAudioFile *)type->tp_alloc(type, 0);
if (self != NULL) {
self->af = new AmAudioFile();
if(!self->af){
Py_DECREF(self);
return NULL;
}
self->py_file = NULL;
#ifdef IVR_WITH_TTS
flite_init();
self->tts_voice = register_cmu_us_kal();
self->filename = new string();
#endif
}
return (PyObject *)self;
}
static void IvrAudioFile_dealloc(IvrAudioFile* self)
{
DBG("---------- IvrAudioFile_dealloc -----------\n");
delete self->af;
self->af = NULL;
// release eventual bound Python file.
Py_XDECREF(self->py_file);
#ifdef IVR_WITH_TTS
if(self->del_file && !self->filename->empty())
unlink(self->filename->c_str());
delete self->filename;
#endif
self->ob_type->tp_free((PyObject*)self);
}
static PyObject* IvrAudioFile_open(IvrAudioFile* self, PyObject* args)
{
int ivr_open_mode;
char* filename;
bool is_tmp;
PyObject* py_is_tmp = NULL;
AmAudioFile::OpenMode open_mode;
if(!PyArg_ParseTuple(args,"si|O",&filename,&ivr_open_mode,&py_is_tmp))
return NULL;
switch(ivr_open_mode){
case AUDIO_READ:
open_mode = AmAudioFile::Read;
break;
case AUDIO_WRITE:
open_mode = AmAudioFile::Write;
break;
default:
PyErr_SetString(PyExc_TypeError,"Unknown open mode");
return NULL;
break;
}
if((py_is_tmp == NULL) || (py_is_tmp == Py_False))
is_tmp = false;
else if(py_is_tmp == Py_True)
is_tmp = true;
else {
PyErr_SetString(PyExc_TypeError,"third parameter should be of type PyBool");
return NULL;
}
if(self->af->open(filename,open_mode,is_tmp)){
PyErr_SetString(PyExc_IOError,"Could not open file");
return NULL;
}
Py_INCREF(Py_None);
return Py_None;
}
static PyObject* IvrAudioFile_fpopen(IvrAudioFile* self, PyObject* args)
{
int ivr_open_mode;
char* filename;
PyObject* py_file = NULL;
AmAudioFile::OpenMode open_mode;
if(!PyArg_ParseTuple(args,"siO",&filename,&ivr_open_mode,&py_file))
return NULL;
switch(ivr_open_mode){
case AUDIO_READ:
open_mode = AmAudioFile::Read;
break;
case AUDIO_WRITE:
open_mode = AmAudioFile::Write;
break;
default:
PyErr_SetString(PyExc_TypeError,"Unknown open mode");
return NULL;
break;
}
FILE* fp = PyFile_AsFile(py_file);
if(!fp){
PyErr_SetString(PyExc_IOError,"Could not get FILE pointer");
return NULL;
}
if(self->af->fpopen(filename,open_mode,fp)){
PyErr_SetString(PyExc_IOError,"Could not open file");
return NULL;
}
// remember, we do not own the file pointer.
// we just want to delay its destruction
self->af->setCloseOnDestroy(false);
self->py_file = py_file;
Py_INCREF(self->py_file);
Py_INCREF(Py_None);
return Py_None;
}
static PyObject* IvrAudioFile_rewind(IvrAudioFile* self, PyObject* args)
{
self->af->rewind();
Py_INCREF(Py_None);
return Py_None;
}
#ifdef IVR_WITH_TTS
static PyObject* IvrAudioFile_tts(PyObject* cls, PyObject* args)
{
char* text;
if(!PyArg_ParseTuple(args,"s",&text))
return NULL;
PyObject* constr_args = Py_BuildValue("(O)",Py_None);
PyObject* tts_file = PyObject_CallObject(cls,constr_args);
Py_DECREF(constr_args);
if(tts_file == NULL){
PyErr_Print();
PyErr_SetString(PyExc_RuntimeError,"could not create new IvrAudioFile object");
return NULL;
}
IvrAudioFile* self = (IvrAudioFile*)tts_file;
*self->filename = string(TTS_CACHE_PATH) + AmSession::getNewId() + string(".wav");
self->del_file = true;
flite_text_to_speech(text,self->tts_voice,self->filename->c_str());
if(self->af->open(self->filename->c_str(),AmAudioFile::Read)){
Py_DECREF(tts_file);
PyErr_SetString(PyExc_IOError,"could not open TTS file");
return NULL;
}
return tts_file;
}
#endif
static PyObject* IvrAudioFile_close(IvrAudioFile* self, PyObject*)
{
self->af->close();
Py_INCREF(Py_None);
return Py_None;
}
static PyObject* IvrAudioFile_getDataSize(IvrAudioFile* self, PyObject*)
{
return PyInt_FromLong(self->af->getDataSize());
}
static PyObject* IvrAudioFile_setRecordTime(IvrAudioFile* self, PyObject* args)
{
int rec_time;
if(!PyArg_ParseTuple(args,"i",&rec_time))
return NULL;
self->af->setRecordTime(rec_time);
Py_INCREF(Py_None);
return Py_None;
}
static PyObject* IvrAudioFile_exportRaw(IvrAudioFile* self, PyObject*)
{
if(self->af->getMode() == AmAudioFile::Write)
self->af->on_close();
self->af->rewind();
return PyFile_FromFile(self->af->getfp(),"","rwb",NULL);
}
static PyMethodDef IvrAudioFile_methods[] = {
{"open", (PyCFunction)IvrAudioFile_open, METH_VARARGS,
"open the audio file"
},
{"fpopen", (PyCFunction)IvrAudioFile_fpopen, METH_VARARGS,
"open the audio file"
},
{"close", (PyCFunction)IvrAudioFile_close, METH_NOARGS,
"close the audio file"
},
{"rewind", (PyCFunction)IvrAudioFile_rewind, METH_NOARGS,
"rewind the audio file"
},
{"getDataSize", (PyCFunction)IvrAudioFile_getDataSize, METH_NOARGS,
"returns the recorded data size"
},
{"setRecordTime", (PyCFunction)IvrAudioFile_setRecordTime, METH_VARARGS,
"set the maximum record time in millisecond"
},
{"exportRaw", (PyCFunction)IvrAudioFile_exportRaw, METH_NOARGS,
"creates a new Python file with the actual file"
" and eventually flushes headers (audio->on_stop)"
},
#ifdef IVR_WITH_TTS
{"tts", (PyCFunction)IvrAudioFile_tts, METH_CLASS | METH_VARARGS,
"text to speech"
},
#endif
{NULL} /* Sentinel */
};
static PyObject* IvrAudioFile_getloop(IvrAudioFile* self, void*)
{
PyObject* loop = self->af->loop.get() ? Py_True : Py_False;
Py_INCREF(loop);
return loop;
}
static int IvrAudioFile_setloop(IvrAudioFile* self, PyObject* value, void*)
{
if (value == NULL) {
PyErr_SetString(PyExc_TypeError, "Cannot delete the first attribute");
return -1;
}
if(value == Py_True)
self->af->loop.set(true);
else if(value == Py_False)
self->af->loop.set(false);
else {
PyErr_SetString(PyExc_TypeError,
"The first attribute value must be a boolean");
return -1;
}
return 0;
}
static PyGetSetDef IvrAudioFile_getseters[] = {
{"loop",
(getter)IvrAudioFile_getloop, (setter)IvrAudioFile_setloop,
"repeat mode",
NULL},
{NULL} /* Sentinel */
};
PyTypeObject IvrAudioFileType = {
PyObject_HEAD_INIT(NULL)
0, /*ob_size*/
"ivr.IvrAudioFile", /*tp_name*/
sizeof(IvrAudioFile), /*tp_basicsize*/
0, /*tp_itemsize*/
(destructor)IvrAudioFile_dealloc, /*tp_dealloc*/
0, /*tp_print*/
0, /*tp_getattr*/
0, /*tp_setattr*/
0, /*tp_compare*/
0, /*tp_repr*/
0, /*tp_as_number*/
0, /*tp_as_sequence*/
0, /*tp_as_mapping*/
0, /*tp_hash */
0, /*tp_call*/
0, /*tp_str*/
0, /*tp_getattro*/
0, /*tp_setattro*/
0, /*tp_as_buffer*/
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /*tp_flags*/
"An audio file", /*tp_doc*/
0, /* tp_traverse */
0, /* tp_clear */
0, /* tp_richcompare */
0, /* tp_weaklistoffset */
0, /* tp_iter */
0, /* tp_iternext */
IvrAudioFile_methods, /* tp_methods */
0, /* tp_members */
IvrAudioFile_getseters, /* tp_getset */
0, /* tp_base */
0, /* tp_dict */
0, /* tp_descr_get */
0, /* tp_descr_set */
0, /* tp_dictoffset */
0, /* tp_init */
0, /* tp_alloc */
IvrAudioFile_new, /* tp_new */
};