mirror of
https://github.com/eeeXun/GTT.git
synced 2025-05-18 08:40:35 -07:00
157 lines
3.9 KiB
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
|
|
}
|