v/GTT
1
0
mirror of https://github.com/eeeXun/GTT.git synced 2025-05-18 08:40:35 -07:00
GTT/internal/translate/reverso/translator.go

157 lines
3.9 KiB
Go

package reverso
import (
"bytes"
"encoding/base64"
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"net/http"
"regexp"
"time"
"github.com/eeeXun/gtt/internal/translate/core"
"github.com/hajimehoshi/go-mp3"
"github.com/hajimehoshi/oto/v2"
)
const (
textURL = "https://api.reverso.net/translate/v1/translation"
ttsURL = "https://voice.reverso.net/RestPronunciation.svc/v1/output=json/GetVoiceStream/voiceName=%s?voiceSpeed=80&inputText=%s"
)
type Translator struct {
*core.APIKey
*core.Language
*core.TTSLock
core.EngineName
}
func NewTranslator() *Translator {
return &Translator{
APIKey: new(core.APIKey),
Language: new(core.Language),
TTSLock: core.NewTTSLock(),
EngineName: core.NewEngineName("Reverso"),
}
}
func (t *Translator) GetAllLang() []string {
return lang
}
func (t *Translator) Translate(message string) (translation *core.Translation, err error) {
translation = new(core.Translation)
var data map[string]interface{}
if t.GetSrcLang() == t.GetDstLang() {
return nil, errors.New(
fmt.Sprintf("%s doesn't support translation of the same language.\ni.e. %s to %s",
t.GetEngineName(), t.GetSrcLang(), t.GetDstLang()))
}
userData, _ := json.Marshal(map[string]interface{}{
"format": "text",
"from": langCode[t.GetSrcLang()],
"to": langCode[t.GetDstLang()],
"input": message,
"options": map[string]string{
"sentenceSplitter": "true",
"origin": "translation.web",
"contextResults": "true",
"languageDetection": "false",
},
})
req, _ := http.NewRequest("POST",
textURL,
bytes.NewBuffer(userData))
req.Header.Add("Content-Type", "application/json")
req.Header.Add("User-Agent", core.UserAgent)
res, err := http.DefaultClient.Do(req)
if err != nil {
return nil, err
}
body, err := ioutil.ReadAll(res.Body)
if err != nil {
return nil, err
}
if err = json.Unmarshal(body, &data); err != nil {
return nil, err
}
if len(data) <= 0 {
return nil, errors.New("Translation not found")
}
// translation
for _, line := range data["translation"].([]interface{}) {
translation.TEXT += line.(string)
}
// definition and part of speech
if data["contextResults"] != nil {
for _, results := range data["contextResults"].(map[string]interface{})["results"].([]interface{}) {
results := results.(map[string]interface{})
// definition
srcExample := results["sourceExamples"].([]interface{})
dstExample := results["targetExamples"].([]interface{})
if len(srcExample) > 0 && len(dstExample) > 0 {
for i := 0; i < len(srcExample) && i < len(dstExample); i++ {
translation.DEF += fmt.Sprintf("- %v\n\t\"%v\"\n", srcExample[i], dstExample[i])
}
}
// part of speech
if results["partOfSpeech"] == nil {
translation.POS += fmt.Sprintf("%v\n", results["translation"])
} else {
translation.POS += fmt.Sprintf("%v [%v]\n", results["translation"], results["partOfSpeech"])
}
}
translation.DEF = regexp.MustCompile("<(|/)em>").ReplaceAllString(translation.DEF, "")
}
return translation, nil
}
func (t *Translator) PlayTTS(lang, message string) error {
defer t.ReleaseLock()
name, ok := voiceName[lang]
if !ok {
return errors.New(t.GetEngineName() + " does not support text to speech of " + lang)
}
urlStr := fmt.Sprintf(
ttsURL,
name,
base64.StdEncoding.EncodeToString([]byte(message)),
)
req, _ := http.NewRequest("GET", urlStr, nil)
req.Header.Add("User-Agent", core.UserAgent)
res, err := http.DefaultClient.Do(req)
if err != nil {
return err
}
decoder, err := mp3.NewDecoder(res.Body)
if err != nil {
return err
}
otoCtx, readyChan, err := oto.NewContext(decoder.SampleRate(), 2, 2)
if err != nil {
return err
}
<-readyChan
player := otoCtx.NewPlayer(decoder)
player.Play()
for player.IsPlaying() {
if t.IsStopped() {
return nil
}
time.Sleep(time.Millisecond)
}
if err = player.Close(); err != nil {
return err
}
return nil
}