You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1009 lines
33 KiB
Go

2 months ago
// go:build !test
//go:build !test
// +build !test
// for STT SELVAS STT
package icsstt
/*
#cgo LDFLAGS: -lstdc++ -ldl ./extlib/selvasstt/SDK/LIB/c_linux/x64/libSTTLIB.a ./extlib/selvasstt/SDK/OpenSSL/Linux/Linux_x64/libssl.a ./extlib/selvasstt/SDK/OpenSSL/Linux/Linux_x64/libcrypto.a
#cgo CFLAGS: -I ../extlib/selvasstt/SDK/INCLUDE
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <STT_Api.h>
char* getResultData(LVCSR_DATA_RESULT* pDataResult, int nCount) {
char* skip1 = "<eps>";
char* space = " ";
int len = 0;
int i;
for (i = 0; i < nCount; i++) {
int maxLength = 50 + strlen(pDataResult[i].pTokenStr);
char* strResult = malloc(sizeof(char) * maxLength);
sprintf(strResult, "%u|%u|%s\n", pDataResult[i].nStart, pDataResult[i].nEnd, pDataResult[i].pTokenStr);
len += strlen(strResult) + 1;
printf("%ld|%ld\n", pDataResult[i].nStart, pDataResult[i].nEnd);
free(strResult);
}
char* result = malloc(sizeof(char) * (len + 50));
strcpy(result, "");
for (i = 0; i < nCount; i++) {
int maxLength = 50 + strlen(pDataResult[i].pTokenStr);
char* strResult = malloc(sizeof(char) * maxLength);
sprintf(strResult, "%u|%u|%s\n", pDataResult[i].nStart, pDataResult[i].nEnd, pDataResult[i].pTokenStr);
if (strcmp(pDataResult[i].pTokenStr,skip1) == 0) {
} else {
strcat(result, space);
strcat(result, strResult);
}
free(strResult);
}
return result;
}
// char* getResultStrData(LVCSR_DATA_RESULT* pResultStr) {
// char* skip1 = "<eps>";
// char* space = " ";
// int len = 0;
// long long i;
// for (i = 0; i < nCount; i++) {
// len += strlen(pDataResult[i].pTokenStr) + 1;
// printf("start: %ld, end: %ld\n", pDataResult[i].nStart, pDataResult[i].nEnd);
// }
// char* result = malloc(sizeof(char) * len);
// strcpy(result, "");
// for (i = 0; i < nCount; i++) {
// if (strcmp(pDataResult[i].pTokenStr,skip1) == 0) {
// } else {
// strcat(result, space);
// strcat(result, pDataResult[i].pTokenStr);
// }
// }
// return result;
// }
void freeResult(char* result){
if(result == NULL) {
return;
}
free(result);
}
int getMidResultEPD(LVCSR_RECOG_MID_RESULT *midResult){
return midResult->bEngineDetectionFlag;
}
void printMidResult(LVCSR_RECOG_MID_RESULT *midResult){
printf("result len : %ld\n", midResult->nResultLen);
printf("pResultStr : %s\n", midResult->pResultStr);
printf("flag : %d\n", midResult->bEngineDetectionFlag);
printf("count: %ld\n\n", midResult->nDataCnt);
}
void setSid(LVCSR_DATA_LOGINFO *logInfo, char* sid, int size) {
// memset(logInfo.tloItemSID, 0, 51);
// memcpy(logInfo.tloItemSID, sid, size);
sprintf(logInfo->tloItemSID, "%s", sid);
// printf("logInfo.tloItemSID-%s\n ", logInfo.tloItemSID);
}
void setCallId(LVCSR_DATA_LOGINFO *logInfo, char* callId, int size) {
// memset(logInfo.tloItemCallID, 0, 61);
// memcpy(logInfo.tloItemCallID, callId, size);
sprintf(logInfo->tloItemCallID, "%s", callId);
// printf("logInfo.tloItemCallID-%s\n ", logInfo.tloItemSID);
}
void setTrId(LVCSR_DATA_LOGINFO *logInfo, char* trId, int size) {
// memset(logInfo.tloItemTransactionID, 0, 101);
// memcpy(logInfo.tloItemTransactionID, trId, size);
sprintf(logInfo->tloItemTransactionID, "%s", trId);
// printf("logInfo.tloItemTransactionID-%s\n ", logInfo.tloItemTransactionID);
}
void setItemStartMessage(LVCSR_DATA_LOGINFO *logInfo, char* tloTime, int size) {
// memset(logInfo.tloItemStartMessage, 0, 101);
// memcpy(logInfo.tloItemStartMessage, trId, size);
sprintf(logInfo->tloItemStartMessage, "%s", tloTime);
// printf("logInfo.tloItemStartMessage-%s\n ", logInfo.tloItemStartMessage);
}
void setConnId(LVCSR_DATA_CALLINFO *callInfo, char* sid, int size) {
callInfo->pConnIdStr = (char*)malloc(size + 1);
memset(callInfo->pConnIdStr, 0, size + 1);
callInfo->nConnIdLen = size;
sprintf(callInfo->pConnIdStr, "%s", sid);
// printf("logInfo.tloItemSID-%s\n ", logInfo.tloItemSID);
}
void setCallDateTime(LVCSR_DATA_CALLINFO *callInfo, unsigned long callDatetime) {
// memset(logInfo.tloItemCallID, 0, 61);
// memcpy(logInfo.tloItemCallID, callId, size);
LVCSR_DATE_TIMESTAMP pCallDateTime = { 0 };
unsigned long sBaseTimeMillis = time(NULL);
SelvySTT_GET_TIMESTAMP(sBaseTimeMillis, &pCallDateTime);
callInfo->pCallDateTime = pCallDateTime;
//sprintf(logInfo->tloItemCallID, "%s", callId);
// printf("logInfo.tloItemCallID-%s\n ", logInfo.tloItemSID);
}
void freeConnId(LVCSR_DATA_CALLINFO *callInfo) {
if (callInfo->pConnIdStr){
free(callInfo->pConnIdStr);
callInfo->pConnIdStr = NULL;
callInfo->nConnIdLen = 0;
}
}
long getSTTSockID(LVCSR_SOCK_HEAD sockhead) {
return sockhead.nSock;
}
*/
import "C"
import (
"fmt"
"log"
"runtime/debug"
"strings"
"sync"
"time"
"unsafe"
// "git.icomsys.co.kr/icomsys/voicecapture/voiceagent/icsconf"
// "git.icomsys.co.kr/icomsys/voicecapture/voiceagent/icserror"
// "git.icomsys.co.kr/icomsys/voicecapture/voiceagent/icslog"
"batchmodule/icsconf"
"batchmodule/icserror"
"batchmodule/icslog"
"github.com/google/uuid"
)
const (
CONNECT_TIMEOUT = 3
READ_CONNECT_TIMEOUT = 5
READ_CONNECT_TIMEOUT_LONGVOICE = 10
// model Info
MODEL_ID = 0
KWD_ID = -1
CODEC_TYPE = 0 // 8k
LANGUAGE = 1 // utf-8
USED_EPD = 1 // epd used
ENGINE_EPD_USED_ON = 2
NO_USED_EPD = 0 // epd used
//USED_EPD = 1 // epd used
USED_SCORE = 0 // used off
USED_ASYNC_MID_RESULT_OFF = 0 // mid result
USED_ASYNC_MID_RESULT_ON = 1 // mid result
DIRECTION_USED_UNK = 0 // Unknown
DIRECTION_USED_INB = 1 // Inbound
DIRECTION_USED_OB = 2 // Outbound
SEG_USED_MONO = 0 // Rx/Tx
SEG_USED_RX = 1 // Receiver
SEG_USED_TX = 2 // Transmitter
SEG_USED_BOT = 3
)
type AsyncCBFunc func(sttResult string, epd int, inout bool)
type STTSelvas struct {
sessionID int
//handle int
authCode string
//ch int
sttID C.long
text string
voiceBuf []byte
voiceBufCur int64
silencenum int
validnum int64 //rms counter
uDataSize int
uEndOfSpeech int
STTInfo STTInfo
IsClose bool
SpeechStatus int
RecordFilePath string
RecordStart *bool
rrData *RRData
M *sync.Mutex
txNum int //number of sending stt audio packet
callID string
custID string
callExt string
sreserved string
trid string
errCode string
errMsg string
errFunName string
result string
reqTime time.Time
rspTime time.Time
svcReqTime time.Time
svcRspTime time.Time
sttIdx int
sttStatus int
//icsstat.StatInfos
}
type STTInfo struct {
LVCSR_SOCK_HEAD C.LVCSR_SOCK_HEAD
LVCSR_EPD_INFO C.LVCSR_EPD_INFO
LVCSR_DATA_AUTHENTICATION C.LVCSR_DATA_AUTHENTICATION
LVCSR_RECOG_RESULT C.LVCSR_RECOG_RESULT
LVCSR_DATA_RESULT C.LVCSR_DATA_RESULT
LVCSR_RECOG_MID_RESULT C.LVCSR_RECOG_MID_RESULT
LVCSR_DATA_INFO C.LVCSR_DATA_INFO
LVCSR_DATA_LOGINFO C.LVCSR_DATA_LOGINFO
LVCSR_ERROR_RESULT C.LVCSR_ERROR_RESULT
LVCSR_DATA_CALLINFO C.LVCSR_DATA_CALLINFO
}
type STTSResult struct {
result string
error *icserror.IcsError
}
var (
conf icsconf.AppInfo
l *log.Logger
)
func init() {
conf = icsconf.Getconfig()
l = icslog.InitializeLogger()
}
func (s STTSelvas) GetTrID() string {
return s.trid
}
// connect SELVAS STT Server
func NewSTTSshort(IP string, port int, connId string, filePath string, isRx bool, isInbound bool) (*STTSelvas, *icserror.IcsError) {
//func NewSTTS(sid int, IP string, port int, callID string, custID string, filePath string, sreserved string, statOK bool) (*STTSelvas, *icserror.IcsError) {
var derr *icserror.IcsError = nil
l.Println("NewSTTSshort() has started!")
defer func() {
if err := recover(); err != nil {
switch v := err.(type) {
case error:
icserror.ICSERRSTTFailInit.SetError(v)
l.Printf("PANIC! %s\n%s\n", icserror.ICSERRSTTFailInit.GetError().Error(), string(debug.Stack()))
default:
l.Println(icserror.ICSERRSTTFailInit.GetError().Error())
}
}
derr = icserror.ICSERRSTTFailInit
}()
if len(IP) <= 0 || port <= 0 {
derr = icserror.ICSERRInvalidParam
return nil, derr
}
// STT 실패 재처리 모듈이라 callId, custId 없어서 connId 넣음.
stts := STTSelvas{authCode: "LGUPlusManager", uDataSize: 1600, uEndOfSpeech: 0, IsClose: false, SpeechStatus: 0, reqTime: time.Now(), svcReqTime: time.Time{}, svcRspTime: time.Time{}, callID: connId, custID: connId, result: "", errCode: "", errMsg: "", errFunName: ""}
//stts := STTSelvas{handle: -1, authCode: "LGUPlusManager", uDataSize: 1600, uEndOfSpeech: 0, IsClose: false}
stts.M = &sync.Mutex{}
stts.txNum = 0
//recording
stts.RecordFilePath = filePath
stts.RecordStart = new(bool)
*stts.RecordStart = true
stts.rrData = NewRRData(960000) //60sec
csIP := C.CString(IP)
defer C.free(unsafe.Pointer(csIP))
csPort := C.int(port)
csConTimeout := C.int(CONNECT_TIMEOUT)
csReadTimeout := C.int(READ_CONNECT_TIMEOUT)
trID := uuid.New().String()
trID = strings.ReplaceAll(trID, "-", "a")
stts.trid = trID
/*
if stts.sttStatus == STTMemo {
return &stts, nil
}
*/
//svcReqTime := time.Now()
l.Println("STT SelvySTT_INIT Start")
rc := C.SelvySTT_INIT(csIP, csPort, csConTimeout, csReadTimeout, &stts.STTInfo.LVCSR_SOCK_HEAD) // add LVCSR_SOCK_HEAD
if int(rc) == -1 {
////////////////테스트코드/////////////////////
rc = C.SelvySTT_GET_ERROR(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_ERROR_RESULT)
if int(rc) < 0 {
l.Printf("[%d] SelvySTT_GET_ERROR Fail.. %d\n", int(stts.sttID), rc)
// return "", derr
} else {
l.Printf("[%d] LVCSR_ERROR_RESULT - [%d]%s\n", int(stts.sttID), stts.STTInfo.LVCSR_ERROR_RESULT.nErrorCode, C.GoString(&stts.STTInfo.LVCSR_ERROR_RESULT.pErrorMsg[0]))
}
l.Printf("STT SelvySTT_INIT Fail - rc:%d\n", rc)
derr = icserror.ICSERRSTTConnectTimeout
return nil, derr
} else if int(rc) == -2 {
l.Printf("STT SelvySTT_INIT Fail - rc:%d\n", rc)
////////////////테스트코드/////////////////////
rc = C.SelvySTT_GET_ERROR(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_ERROR_RESULT)
if int(rc) < 0 {
l.Printf("[%d] SelvySTT_GET_ERROR Fail.. %d\n", int(stts.sttID), rc)
// return "", derr
} else {
l.Printf("[%d] LVCSR_ERROR_RESULT - [%d]%s \n", int(stts.sttID), stts.STTInfo.LVCSR_ERROR_RESULT.nErrorCode, C.GoString(&stts.STTInfo.LVCSR_ERROR_RESULT.pErrorMsg[0]))
}
///////////////////////////////////////////////
derr = icserror.ICSERRSTTFailInit
return nil, derr
}
// callID = strings.ReplaceAll(callID, "-", "")
// if len(callID) > 12 {
// callID = callID[:12]
// }
// l.Printf("callID: %s\n", callID)
// if len(callID) == 0 {
// callID = trID
// }
now := time.Now()
tloTime := fmt.Sprintf("%d%02d%02d%02d%02d%02d\n", now.Year(), int(now.Month()), now.Day(), now.Hour(), now.Minute(), now.Minute())
csTrId := C.CString(trID)
defer C.free(unsafe.Pointer(csTrId))
csConnId := C.CString(connId)
defer C.free(unsafe.Pointer(csConnId))
csTloTime := C.CString(tloTime)
defer C.free(unsafe.Pointer(csTloTime))
// csCustID := C.CString(custID)
csCustID := C.CString(connId)
defer C.free(unsafe.Pointer(csCustID))
// connid 없어서 callid로 connid 만들었던 코드. 해당파일에서는 connId 넣어줌
// C.setConnId(&stts.STTInfo.LVCSR_DATA_CALLINFO, csCallId, C.int(len(callID)))
C.setConnId(&stts.STTInfo.LVCSR_DATA_CALLINFO, csConnId, C.int(len(connId)))
C.setCallDateTime(&stts.STTInfo.LVCSR_DATA_CALLINFO, C.ulong(time.Now().UnixNano()))
if isInbound {
stts.STTInfo.LVCSR_DATA_CALLINFO.nCallDirection = DIRECTION_USED_INB
} else {
stts.STTInfo.LVCSR_DATA_CALLINFO.nCallDirection = DIRECTION_USED_OB
}
if isRx {
stts.STTInfo.LVCSR_DATA_CALLINFO.nSpeakerSegmentation = SEG_USED_RX
} else {
stts.STTInfo.LVCSR_DATA_CALLINFO.nSpeakerSegmentation = SEG_USED_TX
}
l.Printf("[%d] SET SelvySTT_SET_CALLINFO > connID:%s, direction:%d, speakerSeg:%d\n", int(stts.sttID), C.GoString(stts.STTInfo.LVCSR_DATA_CALLINFO.pConnIdStr), stts.STTInfo.LVCSR_DATA_CALLINFO.nCallDirection, stts.STTInfo.LVCSR_DATA_CALLINFO.nSpeakerSegmentation)
// 콜인포 설정
rs := C.SelvySTT_SET_CALLINFO(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_DATA_CALLINFO)
// rs := C.SelvySTT_SET_LOGINFO(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_DATA_LOGINFO)
//ljhwan
//rs = 0
if int(rs) < 0 {
l.Printf("[%d] STT SelvySTT_SET_LOGINFO Fail Result rs:%d, LVCSR_DATA_LOGINFO: %+v\n", int(stts.sttID), rs, stts.STTInfo.LVCSR_DATA_LOGINFO)
////////////////테스트코드/////////////////////
rc = C.SelvySTT_GET_ERROR(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_ERROR_RESULT)
if int(rc) < 0 {
l.Printf("[%d] SelvySTT_GET_ERROR Fail..", int(stts.sttID))
// return "", derr
} else {
l.Printf("[%d] LVCSR_ERROR_RESULT - [%d]%s\n", int(stts.sttID), stts.STTInfo.LVCSR_ERROR_RESULT.nErrorCode, C.GoString(&stts.STTInfo.LVCSR_ERROR_RESULT.pErrorMsg[0]))
}
///////////////////////////////////////////////
// Server Close
//svcReqTime = time.Now()
l.Printf("[%d] STT SelvySTT_EXIT Start\n", int(stts.sttID))
rc = C.SelvySTT_EXIT(&stts.STTInfo.LVCSR_SOCK_HEAD)
if int(rc) < 0 {
l.Printf("[%d] STT SelvySTT_EXIT Fail Result rs:%d\n", int(stts.sttID), rs)
derr = icserror.ICSERRSTTFailInit
return nil, derr
}
derr = icserror.ICSERRSTTFailInit
return nil, derr
}
C.freeConnId(&stts.STTInfo.LVCSR_DATA_CALLINFO)
//svcReqTime = time.Now()
// Set Model List
stts.STTInfo.LVCSR_DATA_INFO.nModelID = MODEL_ID
stts.STTInfo.LVCSR_DATA_INFO.nKwdID = KWD_ID
stts.STTInfo.LVCSR_DATA_INFO.nCodecType = CODEC_TYPE
stts.STTInfo.LVCSR_DATA_INFO.nCharSet = LANGUAGE
// if stts.sttStatus == STTMemo {
// stts.STTInfo.LVCSR_DATA_INFO.nEpdUsed = NO_USED_EPD
// } else {
stts.STTInfo.LVCSR_DATA_INFO.nEpdUsed = ENGINE_EPD_USED_ON
stts.STTInfo.LVCSR_DATA_INFO.bAsyncMidRstUsed = USED_ASYNC_MID_RESULT_ON
// }
stts.STTInfo.LVCSR_DATA_INFO.bScoreUsed = USED_SCORE
// Channel Connect
//svcReqTime = time.Now()
l.Printf("[%d] STT SelvySTT_OPEN Start\n", int(stts.sttID))
rc = C.SelvySTT_OPEN(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_DATA_INFO) // add LVCSR_SOCK_HEAD
//ljhwan
//rc = 0
if int(rc) < 0 {
l.Printf("[%d] STT SelvySTT_OPEN Fail, LVCSR_SOCK_HEAD: %+v\r\n", int(stts.sttID), stts.STTInfo.LVCSR_SOCK_HEAD)
// Server Close
//svcReqTime = time.Now()
l.Printf("[%d] STT SelvySTT_EXIT Start", int(stts.sttID))
rc = C.SelvySTT_EXIT(&stts.STTInfo.LVCSR_SOCK_HEAD)
if int(rc) < 0 {
l.Printf("[%d]STT SelvySTT_EXIT Fail Result rs:%d, LVCSR_SOCK_HEAD: %+v\n", int(stts.sttID), rs, stts.STTInfo.LVCSR_SOCK_HEAD)
derr = icserror.ICSERRSTTFailInit
return nil, derr
}
derr = icserror.ICSERRSTTFailInit
return nil, derr
}
// rc = C.SelvySTT_SET_INFO(&stts.STTInfo.LVCSR_SOCK_HEAD, &stts.STTInfo.LVCSR_DATA_INFO)
// //ljhwan
// //rc = 0
// if int(rc) < 0 {
// l.Printf("[%d]STT SelvySTT_SET_INFO Fail Result rs:%d, LVCSR_SOCK_HEAD: %+v, stts.STTInfo.LVCSR_DATA_INFO: %+v\n", int(stts.sttID), rs, stts.STTInfo.LVCSR_SOCK_HEAD, stts.STTInfo.LVCSR_DATA_INFO)
// // Channel Connection Close
// //svcReqTime = time.Now()
// l.Printf("[%d]STT SelvySTT_CLOS Start\n", int(stts.sttID))
// rc = C.SelvySTT_CLOS(&stts.STTInfo.LVCSR_SOCK_HEAD)
// if int(rc) < 0 {
// l.Printf("[%d]STT SelvySTT_CLOS Fail Result rs:%d, LVCSR_SOCK_HEAD: %+v\n", int(stts.sttID), rs, stts.STTInfo.LVCSR_SOCK_HEAD)
// derr = icserror.ICSERRTTSNotInit
// }
// // Server Close
// l.Printf("[%d]STT SelvySTT_EXIT Start\n", int(stts.sttID))
// //svcReqTime = time.Now()
// rc = C.SelvySTT_EXIT(&stts.STTInfo.LVCSR_SOCK_HEAD)
// if int(rc) < 0 {
// l.Printf("[%d]STT SelvySTT_EXIT Fail Result rs:%d, LVCSR_SOCK_HEAD: %+v\n", int(stts.sttID), rs, stts.STTInfo.LVCSR_SOCK_HEAD)
// derr = icserror.ICSERRSTTFailInit
// return nil, derr
// }
// derr = icserror.ICSERRSTTFailInit
// return nil, derr
// }
//l.Printf("[%d]Selvas STT New Connection > callID:'%s',trID:'%s'\n", int(stts.sttID), callID, trID)
l.Printf("[%d]Selvas STT New Connection > connId:'%s',trID:'%s'\n", int(stts.sttID), connId, trID)
return &stts, derr
}
func (s *STTSelvas) Close() *icserror.IcsError {
if s == nil {
return icserror.ICSERRSTTNotInit
}
var derr *icserror.IcsError = nil
defer func() {
if err := recover(); err != nil {
switch v := err.(type) {
case error:
icserror.ICSERRSTTFailInit.SetError(v)
l.Printf("PANIC! %s\n%s\n", icserror.ICSERRSTTFailInit.GetError().Error(), string(debug.Stack()))
default:
l.Print(icserror.ICSERRSTTFailInit.GetError().Error())
}
derr = icserror.ICSERRSTTFailInit
return
}
// derr = icserror.ICSERRSTTFailInit
}()
l.Printf("Close STT[%d]. Send Audio Packet Num: %d\n", int(s.sttID), s.txNum)
s.M.Lock()
s.IsClose = true
s.M.Unlock()
*s.RecordStart = false
rc := C.SelvySTT_CLS_RES(&s.STTInfo.LVCSR_RECOG_RESULT)
if int(rc) < 0 {
l.Printf("Failed to free STT result[%d]. rc: %d\n", int(s.sttID), rc)
if strings.Compare(s.errCode, "54000013") != 0 {
s.errCode, s.errMsg, s.errFunName = "54000010", "Fail", "SelvySTT_CLS_RES"
}
derr = icserror.ICSERRSTTFreeError
}
// Channel Connection Close
l.Printf("Close STT channel[%d]\n", int(s.sttID))
rc = C.SelvySTT_CLOS(&s.STTInfo.LVCSR_SOCK_HEAD)
if int(rc) < 0 {
l.Printf("Failed to close STT channel[%d], rc: %d\n", int(s.sttID), rc)
if s.result == "$NO_RESULT$" {
s.errCode, s.errMsg, s.errFunName = "20000003", "Success", ""
} else if strings.Compare(s.errCode, "54000013") != 0 {
s.errCode, s.errMsg, s.errFunName = "54000011", "Fail", "SelvySTT_CLOS"
}
derr = icserror.ICSERRTTSNotInit
}
// Server Close
l.Printf("Close STT SVC[%d]\n", int(s.sttID))
rc = C.SelvySTT_EXIT(&s.STTInfo.LVCSR_SOCK_HEAD)
//ljhwan
//rc = 0
if int(rc) < 0 {
l.Printf("Failed to close STT SVC[%d], rc: %d\n", int(s.sttID), rc)
derr = icserror.ICSERRTTSNotInit
if strings.Compare(s.errCode, "54000013") != 0 {
s.errCode, s.errMsg, s.errFunName = "54000012", "Fail", "SelvySTT_EXIT"
}
}
l.Printf("Closed STT[%d] sock header: %+v\n", int(s.sttID), s.STTInfo.LVCSR_SOCK_HEAD.nSock)
s.rrData.Clear()
return derr
}
func (s *STTSelvas) GetSTTStatus() bool {
if s == nil {
// fmt.Println("get status: stt handle nil")
return false
}
s.M.Lock()
b := s.IsClose
s.M.Unlock()
return b
}
// type AsyncCBFunc func(sttResult string, epd int)
func (s *STTSelvas) SendSTT(voicedata []byte, final bool, cbFunc AsyncCBFunc) (string, *icserror.IcsError) {
if s == nil {
return "", icserror.ICSERRTTSNotInit
}
var result string
var sendCount int
var derr *icserror.IcsError = nil
var resultsWithTime string
defer func() {
if err := recover(); err != nil {
switch v := err.(type) {
case error:
icserror.ICSERRSTTSendFail.SetError(v)
l.Printf("sttselvas.go>SendSTT()> PANIC! %s\n%s\n", icserror.ICSERRSTTSendFail.GetError().Error(), string(debug.Stack()))
default:
l.Print(icserror.ICSERRSTTSendFail.GetError().Error())
}
}
derr = icserror.ICSERRSTTSendFail
}()
// if s.sttStatus != STTMemo {
// Set EPD value
if !final {
s.uEndOfSpeech = 0
} else {
s.uEndOfSpeech = 1
}
// file split
// sendCount = len(voicedata) / 320
sendCount = len(voicedata) / 1600
// l.Printf("SendSTT voice size: %d, sendCount: %d \n", len(voicedata), sendCount)
//start := time.Now()
// send voice(voicedata)
for j := 0; j < sendCount; j++ {
// buff1 := voicedata[320*j : 320*(j+1)]
buff1 := voicedata[1600*j : 1600*(j+1)]
//recording audio
if *s.RecordStart {
s.rrData.AddAudio(buff1)
} else {
s.rrData.AddTempAudio(buff1)
}
csUDataSize := C.int(s.uDataSize)
// if !final && j < sendCount-1 {
// s.uEndOfSpeech = 0
// } else {
// s.uEndOfSpeech = 1
// }
csUEndSpeech := C.int(s.uEndOfSpeech)
csBuff := (*C.char)(unsafe.Pointer(&buff1[0]))
// l.Printf("SelvySTT_SEND_DATA : %d, size: %d EndofSpeech : %d\n", j, csUDataSize, s.uEndOfSpeech)
rc := C.SelvySTT_SEND_DATA(&s.STTInfo.LVCSR_SOCK_HEAD,
csBuff,
csUDataSize,
csUEndSpeech,
&s.STTInfo.LVCSR_EPD_INFO)
if int(rc) < 0 {
l.Printf("sttselvas.go>SendSTT()> [%d] SelvySTT_SEND_DATA ERROR %d\n", int(s.sttID), rc)
////////////////테스트코드/////////////////////
rc = C.SelvySTT_GET_ERROR(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_ERROR_RESULT)
if int(rc) < 0 {
l.Printf("sttselvas.go>SendSTT()> [%d] SelvySTT_GET_ERROR Fail.. %d\n", int(s.sttID), rc)
// return "", derr
} else {
l.Printf("sttselvas.go>SendSTT()> [%d] LVCSR_ERROR_RESULT - [%d]%s\n", int(s.sttID), s.STTInfo.LVCSR_ERROR_RESULT.nErrorCode, C.GoString(&s.STTInfo.LVCSR_ERROR_RESULT.pErrorMsg[0]))
}
///////////////////////////////////////////////
s.errCode, s.errMsg, s.errFunName = "54000013", "Fail", "SelvySTT_SEND_DATA"
if cbFunc != nil {
cbFunc("", 99, true)
break
}
derr = icserror.ICSERRSTTSendFail
return "", derr
}
s.txNum++
if s.STTInfo.LVCSR_EPD_INFO == 1 {
//l.Printf("STT ID[%d] LVCSR_EPD_INFO: %d", int(s.sttID), s.STTInfo.LVCSR_EPD_INFO)
if s.SpeechStatus == 0 {
l.Printf("sttselvas.go>SendSTT()> STT ID[%d] LVCSR_EPD_INFO: %d\n", int(s.sttID), s.STTInfo.LVCSR_EPD_INFO)
s.SpeechStatus = 1
if cbFunc != nil {
cbFunc(result, int(s.STTInfo.LVCSR_EPD_INFO), true)
}
}
} else if s.STTInfo.LVCSR_EPD_INFO > 1 {
l.Printf("sttselvas.go>SendSTT()> LVCSR_EPD_INFO == %d\n", s.STTInfo.LVCSR_EPD_INFO)
s.svcReqTime = time.Now()
break
}
}
if s.STTInfo.LVCSR_EPD_INFO == 2 || s.STTInfo.LVCSR_EPD_INFO == 5 {
// l.Printf("[%d]!!!! LVCSR_EPD_INFO: %d, LVCSR_SOCK_HEAD: %+v", int(s.sttID), s.STTInfo.LVCSR_EPD_INFO, s.STTInfo.LVCSR_SOCK_HEAD)
l.Printf("sttselvas.go>SendSTT()> STT ID[%d] LVCSR_EPD_INFO: %d, STT Completed!\n", int(s.sttID), s.STTInfo.LVCSR_EPD_INFO)
rc := C.SelvySTT_GET_MIDRES(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_MID_RESULT)
s.svcRspTime = time.Now()
if int(rc) < 0 {
l.Printf("sttselvas.go>SendSTT()> [%d]SelvySTT_GET_MIDRES Failed!!! - rc:%d, s.STTInfo.LVCSR_RECOG_MID_RESULT:%+v\n", int(s.sttID), rc, s.STTInfo.LVCSR_RECOG_MID_RESULT)
// s.errCode, s.errMsg, s.errFunName = "54000009", "Fail", "ASR_SVC_RECG_STR_PROC"
s.result = "$NO_RESULT$"
derr = icserror.ICSERRSTTSendFail
if cbFunc != nil {
cbFunc("", 2, true)
}
return "", icserror.ICSERRSTTContinue // 묵음으로 처리
}
s.sttIdx++
wg := sync.WaitGroup{}
wg.Add(1)
go func() {
result = C.GoString(s.STTInfo.LVCSR_RECOG_MID_RESULT.pResultStr)
resultTocken := C.GoString(s.STTInfo.LVCSR_RECOG_MID_RESULT.pDataResult.pTokenStr)
nStartTime := C.int(s.STTInfo.LVCSR_RECOG_MID_RESULT.pDataResult.nStart)
nEnd := C.int(s.STTInfo.LVCSR_RECOG_MID_RESULT.pDataResult.nEnd)
l.Printf("sttselvas.go>SendSTT()> STT ID[%d] result: %s, result: %p, pointer: %p\n", int(s.sttID), result, &result, s.STTInfo.LVCSR_RECOG_RESULT.pResultStr)
l.Printf("sttselvas.go>SendSTT()> STT ID[%d] result: %s, %d, %d\n", int(s.sttID), resultTocken, nStartTime, nEnd)
// l.Printf("STT ID[%d] result: %s, result: %p, pointer: %p\n", int(s.sttID), result, &result, s.STTInfo.LVCSR_RECOG_RESULT.pResultStr)
results := ""
if s.STTInfo.LVCSR_RECOG_MID_RESULT.nResultLen == 0 { // 길이 0일때 값 nil로 나옴
result = ""
}
//result = "TEST TEST TEST TEST TEST TEST TEST TEST TEST TEST TEST"
s.result = result
results = result
if result == "$NO_RESULT$" {
results = ""
}
resultsWithTime = fmt.Sprintf("%d|%d|%s\n", nStartTime, nEnd, results)
if cbFunc != nil {
cbFunc(results, int(s.STTInfo.LVCSR_EPD_INFO), true)
}
//l.Printf("[%d] result Free OK", int(s.sttID))
wg.Done()
}()
wg.Wait()
// 중간결과 구조체 메모리 해제
rc = C.SelvySTT_CLS_MIDRES(&s.STTInfo.LVCSR_RECOG_MID_RESULT)
if int(rc) < 0 {
l.Println("sttselvas.go>SendSTT()> SelvySTT_CLS_MIDRES Failed!!! MID RES pResultStr: ", s.STTInfo.LVCSR_RECOG_MID_RESULT.pResultStr)
}
return resultsWithTime, icserror.ICSERRSTTContinue
}
// } else {
// s.uEndOfSpeech = 1
// //recording audio
// if *s.RecordStart {
// s.rrData.AddAudio(voicedata)
// } else {
// s.rrData.AddTempAudio(voicedata)
// }
// csUDataSize := C.int(s.uDataSize)
// csUEndSpeech := C.int(s.uEndOfSpeech)
// csBuff := (*C.char)(unsafe.Pointer(&voicedata))
// //csBuff := (*C.char)(unsafe.Pointer(&buff1[0]))
// rc := C.SelvySTT_SEND_DATA(&s.STTInfo.LVCSR_SOCK_HEAD,
// csBuff,
// csUDataSize,
// csUEndSpeech,
// &s.STTInfo.LVCSR_EPD_INFO)
// if int(rc) < 0 {
// l.Printf("[%d] SelvySTT_SEND_DATA ERROR %d", int(s.sttID), rc)
// ////////////////테스트코드/////////////////////
// rc = C.SelvySTT_GET_ERROR(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_ERROR_RESULT)
// if int(rc) < 0 {
// l.Printf("[%d] SelvySTT_GET_ERROR Fail.. %d", int(s.sttID), rc)
// // return "", derr
// } else {
// l.Printf("[%d] LVCSR_ERROR_RESULT - [%d]%s", int(s.sttID), s.STTInfo.LVCSR_ERROR_RESULT.nErrorCode, C.GoString(&s.STTInfo.LVCSR_ERROR_RESULT.pErrorMsg[0]))
// }
// ///////////////////////////////////////////////
// s.errCode, s.errMsg, s.errFunName = "54000013", "Fail", "SelvySTT_SEND_DATA"
// if cbFunc != nil {
// cbFunc("", 99, true)
// }
// derr = icserror.ICSERRSTTSendFail
// return "", derr
// }
// s.txNum++
// rc = C.SelvySTT_GET_MIDRES(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_MID_RESULT)
// s.svcRspTime = time.Now()
// if int(rc) < 0 {
// l.Printf("[%d]!!!! SelvySTT_GET_MIDRES Fail - rc:%d, s.STTInfo.LVCSR_RECOG_MID_RESULT:%+v", int(s.sttID), rc, s.STTInfo.LVCSR_RECOG_MID_RESULT)
// // s.errCode, s.errMsg, s.errFunName = "54000009", "Fail", "ASR_SVC_RECG_STR_PROC"
// s.result = "$NO_RESULT$"
// derr = icserror.ICSERRSTTSendFail
// if cbFunc != nil {
// cbFunc("", 2, true)
// }
// return "", icserror.ICSERRSTTContinue // 묵음으로 처리
// }
// result = C.GoString(s.STTInfo.LVCSR_RECOG_RESULT.pResultStr)
// l.Printf("STT ID[%d] result: %s, result: %p, pointer: %p", int(s.sttID), result, &result, s.STTInfo.LVCSR_RECOG_RESULT.pResultStr)
// results := ""
// if s.STTInfo.LVCSR_RECOG_RESULT.nResultLen == 0 { // 길이 0일때 값 nil로 나옴
// result = ""
// }
// //result = "TEST TEST TEST TEST TEST TEST TEST TEST TEST TEST TEST"
// s.result = result
// results = result
// if result == "$NO_RESULT$" {
// results = ""
// }
// if cbFunc != nil {
// cbFunc(results, int(s.STTInfo.LVCSR_EPD_INFO), true)
// }
// //l.Printf("[%d] result Free OK", int(s.sttID))
// return results, icserror.ICSERRSTTOK
// }
return "", icserror.ICSERRSTTContinue
}
// result return
func NewSTTSResult(result string, err *icserror.IcsError) *STTSResult {
return &STTSResult{result, err}
}
// 말을 하지 않았을때 결과 조회
func (s *STTSelvas) GetSTTResultToStopSTT() (string, *icserror.IcsError) {
l := icslog.GetIcsLog()
if s == nil {
return "", icserror.ICSERRSTTNotInit
}
//rc := C.SelvySTT_GET_MIDRES(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_MID_RESULT)
rc := C.SelvySTT_GET_RES(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_RESULT)
l.Printf("GetSTTResultToStopSTT rc:%d, [%d] EPDINFO[%d] LVCSR_RECOG_RESULT: %+v, LVCSR_RECOG_RESULT:%+v\n", int(rc), int(s.sttID), s.STTInfo.LVCSR_EPD_INFO, &s.STTInfo.LVCSR_RECOG_RESULT, s.STTInfo.LVCSR_RECOG_RESULT)
if int(rc) < 0 {
l.Printf("[%d] SelvySTT_GET_MIDRES Fail - rc:%d, s.STTInfo.LVCSR_RECOG_MID_RESULT:%+v\n", int(s.sttID), rc, s.STTInfo.LVCSR_RECOG_MID_RESULT)
s.errCode, s.errMsg, s.errFunName = "54000009", "Fail", "SelvySTT_GET_MIDRES"
// s.SendStasticInfo(s.callID, s.custID, s.trid, "STT", "ASR_SVC_RECG_STR_PROC", "54000001", "Fail", "", "", s.reqTime, time.Now(), s.sreserved, svcReqTime, time.Now())
return "", icserror.ICSERRSTTSendFail
}
return "", icserror.ICSERRSTTContinue
}
// epd 1 이후 end 전송 => 결과조회
func (s *STTSelvas) SendSTTProcGetResult() (string, *icserror.IcsError) {
var derr *icserror.IcsError = nil
if s == nil {
return "", icserror.ICSERRSTTNotInit
}
//buff1 := make([]byte, 1)
s.uEndOfSpeech = 1
uDataSize := 0
csUDataSize := C.int(uDataSize)
csUEndSpeech := C.int(s.uEndOfSpeech)
//csBuff := (*C.char)(unsafe.Pointer(&buff1[0]))
// csBuff := nil
// svcReqTime := time.Now()
rc := C.SelvySTT_SEND_DATA(&s.STTInfo.LVCSR_SOCK_HEAD,
nil,
csUDataSize,
csUEndSpeech,
&s.STTInfo.LVCSR_EPD_INFO)
if int(rc) < 0 {
l.Printf("[ERR] sttselvas>SendSTTProcGetResult()> [%d] SelvySTT_SEND_DATA Failed - rc:%d, s.STTInfo.LVCSR_RECOG_RESULT:%+v\n", int(s.sttID), rc, s.STTInfo.LVCSR_RECOG_RESULT)
derr = icserror.ICSERRSTTSendFail
s.errCode, s.errMsg, s.errFunName = "54000009", "Fail", "SelvySTT_SEND_DATA"
// s.SendStasticInfo(s.callID, s.custID, s.trid, "STT", "SelvySTT_SEND_DATA", "54000001", "Fail", "", "", s.reqTime, time.Now(), s.sreserved, svcReqTime, time.Now())
return "", derr
}
l.Printf("[LOG] sttselvas>SendSTTProcGetResult()> rc:%d, [%d] EPDINFO[%d] LVCSR_RECOG_RESULT: %+v, LVCSR_RECOG_RESULT:%+v\n",
int(rc), int(s.sttID), s.STTInfo.LVCSR_EPD_INFO, &s.STTInfo.LVCSR_RECOG_RESULT, s.STTInfo.LVCSR_RECOG_RESULT)
if s.STTInfo.LVCSR_EPD_INFO == 2 {
// l.Printf("[%d]!!!! LVCSR_EPD_INFO: %d, LVCSR_SOCK_HEAD: %+v", int(s.sttID), s.STTInfo.LVCSR_EPD_INFO, s.STTInfo.LVCSR_SOCK_HEAD)
l.Printf("[LOG] sttselvas>SendSTTProcGetResult()> STT ID[%d] LVCSR_EPD_INFO: %d, STT Completed!\n", int(s.sttID), s.STTInfo.LVCSR_EPD_INFO)
rc := C.SelvySTT_GET_MIDRES(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_MID_RESULT)
s.svcRspTime = time.Now()
if int(rc) < 0 {
l.Printf("[ERR] sttselvas>SendSTTProcGetResult()> [%d] SelvySTT_GET_MIDRES Failed - rc:%d, s.STTInfo.LVCSR_RECOG_MID_RESULT:%+v\n", int(s.sttID), rc, s.STTInfo.LVCSR_RECOG_MID_RESULT)
// s.errCode, s.errMsg, s.errFunName = "54000009", "Fail", "ASR_SVC_RECG_STR_PROC"
s.result = "$NO_RESULT$"
// derr = icserror.ICSERRSTTSendFail
return "", icserror.ICSERRSTTContinue // 묵음으로 처리
}
wg := sync.WaitGroup{}
wg.Add(1)
go func() {
result := C.GoString(s.STTInfo.LVCSR_RECOG_MID_RESULT.pResultStr)
l.Printf("[LOG] sttselvas>SendSTTProcGetResult()> STT ID[%d] result: %s, result: %p, pointer: %p\n", int(s.sttID), result, &result, s.STTInfo.LVCSR_RECOG_RESULT.pResultStr)
results := ""
if s.STTInfo.LVCSR_RECOG_MID_RESULT.nResultLen == 0 { // 길이 0일때 값 nil로 나옴
result = ""
}
//result = "TEST TEST TEST TEST TEST TEST TEST TEST TEST TEST TEST"
s.result = result
results = result
if result == "$NO_RESULT$" {
results = ""
}
l.Printf("[LOG] sttselvas>SendSTTProcGetResult()> results: %s", results)
//l.Printf("[%d] result Free OK", int(s.sttID))
wg.Done()
}()
wg.Wait()
}
// 중간결과 구조체 메모리 해제
rc = C.SelvySTT_CLS_MIDRES(&s.STTInfo.LVCSR_RECOG_MID_RESULT)
if int(rc) < 0 {
l.Println("[ERR] sttselvas>SendSTTProcGetResult()> SelvySTT_CLS_MIDRES Failed. LVCSR_RECOG_MID_RESULT.pResultStr: ", s.STTInfo.LVCSR_RECOG_MID_RESULT.pResultStr)
}
// 최종 결과 호출
rc = C.SelvySTT_GET_RES(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_RESULT)
if int(rc) < 0 {
l.Printf("[ERR] sttselvas>SendSTTProcGetResult()> [%d] SelvySTT_GET_RES Fail - rc:%d, s.STTInfo.SelvySTT_GET_RES:%+v\n", int(s.sttID), rc, s.STTInfo.LVCSR_RECOG_RESULT)
derr = icserror.ICSERRSTTSendFail
s.errCode, s.errMsg, s.errFunName = "54000009", "Fail", "SelvySTT_GET_RES"
return "", derr
}
dataCnt := int(s.STTInfo.LVCSR_RECOG_RESULT.nDataCnt)
resultLen := int(s.STTInfo.LVCSR_RECOG_RESULT.nResultLen)
pResultStr := C.GoString(s.STTInfo.LVCSR_RECOG_RESULT.pResultStr)
l.Printf("dataCnt: %d", dataCnt)
l.Printf("resultLen: %d", resultLen)
if pResultStr != "" {
strresult := C.getResultData(s.STTInfo.LVCSR_RECOG_RESULT.pDataResult, C.int(dataCnt))
l.Printf("[LOG] sttselvas>SendSTTProcGetResult()> SelvySTT_GET_RES 인식 결과: %s", C.GoString(strresult))
defer C.free(unsafe.Pointer(strresult))
if resultLen > 0 {
l.Printf("[LOG] sttselvas>SendSTTProcGetResult()> SelvySTT_GET_RES 인식 결과: 연속어 [ %s ], 첫음 [ %d ] 끝음 [ %d ] 스코어 [ %f ]",
pResultStr, s.STTInfo.LVCSR_RECOG_RESULT.pDataResult.nStart, s.STTInfo.LVCSR_RECOG_RESULT.pDataResult.nEnd, s.STTInfo.LVCSR_RECOG_RESULT.fConfidScore)
} else {
return "", icserror.ICSERRSTTFail
}
// if dataCnt > 0 {
// for i := 0; i < dataCnt; i++ {
// l.Printf(icslog.LOG_LEVEL_INFO, -1, "음성인식 결과 : 인식어 [ %s ], 시작시간 [ %d ] 종료시간 [ %d ]", s.STTInfo.LVCSR_RECOG_RESULT.pDataResult[i].pTokenStr, s.STTInfo.LVCSR_RECOG_RESULT.pDataResult[i].nStart, s.STTInfo.LVCSR_RECOG_RESULT.pDataResult[i].nEnd)
// }
// }
//s.GetSTTResultToStopSTT()
// fmt.Print("Request Recg Result")
// rc = C.ASR_SVC_RECG_PROC(&s.STTInfo.LVCSR_SOCK_HEAD, &s.STTInfo.LVCSR_RECOG_RESULT)
// if int(rc) < 0 {
// derr = icserror.ICSERRSTTFail
// return "", derr
// }
returnstr := C.GoString(strresult)
return returnstr, nil
}
return "", nil
}
func long(dataCnt int) {
panic("unimplemented")
}
// func getResultMessage(result *C.LVCSR_RECOG_RESULT, size int) []string {
// messages := make([]string, size)
// for i := 0; i < size; i++ {
// cMsg := C.GoString(result.pDataResult)
// }
// return s.rrData.GetAudio()
// }
func (s STTSelvas) GetAudio() []byte {
return s.rrData.GetAudio()
}
func (s STTSelvas) GetReqTime() time.Time {
return s.reqTime
}