refactor: use online query method template instead of backend.query todo online query

This commit is contained in:
JuanZoran 2023-03-18 21:31:14 +08:00
parent b2cafe3448
commit 52d2741804
7 changed files with 119 additions and 205 deletions

View File

@ -1,16 +1,17 @@
---@class Baidu: TransBackend ---@class Baidu: TransOnlineBackend
---@field uri string api uri ---@field uri string api uri
---@field salt string ---@field salt string
---@field app_id string ---@field app_id string
---@field app_passwd string ---@field app_passwd string
---@field disable boolean ---@field disable boolean
local M = { local M = {
uri = "https://fanyi-api.baidu.com/api/trans/vip/translate", uri = "https://fanyi-api.baidu.com/api/trans/vip/translate",
salt = tostring(math.random(bit.lshift(1, 15))), salt = tostring(math.random(bit.lshift(1, 15))),
name = "baidu", name = "baidu",
method = 'get',
} }
local Trans = require("Trans") local Trans = require("Trans")
---@class BaiduQuery ---@class BaiduQuery
@ -24,7 +25,7 @@ local Trans = require("Trans")
---Get content for query ---Get content for query
---@param data TransData ---@param data TransData
---@return BaiduQuery ---@return BaiduQuery
function M.get_content(data) function M.get_query(data)
local tmp = M.app_id .. data.str .. M.salt .. M.app_passwd local tmp = M.app_id .. data.str .. M.salt .. M.app_passwd
local sign = Trans.util.md5.sumhexa(tmp) local sign = Trans.util.md5.sumhexa(tmp)
@ -38,42 +39,21 @@ function M.get_content(data)
} }
end end
-- {
-- body = '{"from":"en","to":"zh","trans_result":[{"src":"require","dst":"\\u8981\\u6c42"}]}',
-- exit = 0,
-- headers = { "Content-Type: application/json", "Date: Thu, 09 Mar 2023 14:01:09 GMT", 'P3p: CP=" OTI DSP COR IVA OUR IND COM "', "Server: Apache", "Set-Cookie: BAIDUID=CB6D99CCD3B5F5278B5BE9428F002FC3:FG=1; expires=Fri, 08-Mar-24 14:01:09 GMT; max-age=31536000; path=/; domain=.baidu.com; version=1", "Tracecode: 00696104432377504778030922", "Content-Length: 79", "", "" },
-- status = 200
-- }
---@overload fun(TransData): TransResult ---@overload fun(TransData): TransResult
---Query Using Baidu API ---Query Using Baidu API
---@param data TransData ---@param body table BaiduQuery Response
function M.query(data) ---@return table|false
local handle = function(res) function M.formatter(body, data)
local status, body = pcall(vim.json.decode, res.body) local result = body.trans_result
if not status or not body then if not result then return false end
data.trace = res
data.result.baidu = false
return
end
local result = body.trans_result -- TEST :whether multi result
assert(#result == 1)
if result then result = result[1]
-- TEST :whether multi result return {
assert(#result == 1) str = result.src,
result = result[1] [data.from == "en" and "translation" or "definition"] = { result.dst },
data.result.baidu = { }
["str"] = result.src,
[data.from == "en" and "translation" or "definition"] = { result.dst },
}
end
end
Trans.curl.get(M.uri, {
query = M.get_content(data),
callback = handle,
})
end end
---@class TransBackend ---@class TransBackend
@ -83,3 +63,10 @@ return M
-- -- NOTE :free tts: -- -- NOTE :free tts:
-- -- https://zj.v.api.aa1.cn/api/baidu-01/?msg=我爱你&choose=0&su=100&yd=5 -- -- https://zj.v.api.aa1.cn/api/baidu-01/?msg=我爱你&choose=0&su=100&yd=5
-- -- 选择转音频的人物女生1 输入0 女生2输入5男生1 输入1男生2 输入2男生3 输入3 -- -- 选择转音频的人物女生1 输入0 女生2输入5男生1 输入1男生2 输入2男生3 输入3
-- {
-- body = '{"from":"en","to":"zh","trans_result":[{"src":"require","dst":"\\u8981\\u6c42"}]}',
-- exit = 0,
-- headers = { "Content-Type: application/json", "Date: Thu, 09 Mar 2023 14:01:09 GMT", 'P3p: CP=" OTI DSP COR IVA OUR IND COM "', "Server: Apache", "Set-Cookie: BAIDUID=CB6D99CCD3B5F5278B5BE9428F002FC3:FG=1; expires=Fri, 08-Mar-24 14:01:09 GMT; max-age=31536000; path=/; domain=.baidu.com; version=1", "Tracecode: 00696104432377504778030922", "Content-Length: 79", "", "" },
-- status = 200
-- }

View File

@ -1,9 +1,4 @@
---@class iCiba: TransBackend ---@class iCiba: TransOnlineBackend
---@field uri string api uri
---@field salt string
---@field app_id string
---@field app_passwd string
---@field disable boolean
local M = { local M = {
uri = 'https://dict-mobile.iciba.com/interface/index.php', uri = 'https://dict-mobile.iciba.com/interface/index.php',
name = 'iciba', name = 'iciba',
@ -16,7 +11,7 @@ local M = {
---@field appid string ---@field appid string
---@field salt string ---@field salt string
---@field sign string ---@field sign string
function M.get_content(data) function M.get_query(data)
return { return {
word = data.str, word = data.str,
is_need_mean = '1', is_need_mean = '1',
@ -25,47 +20,14 @@ function M.get_content(data)
} }
end end
---@overload fun(TransData): TransResult
function M.query(data)
local handle = function(res)
local status, body = pcall(vim.json.decode, res.body)
vim.print(body)
if true and not status or not body or body.errorCode ~= "0" then function M.formatter(body, data)
data.result.iciba = false print('TODO')
data[#data + 1] = res -- if true and not status or not body or body.errorCode ~= "0" then
return -- data.result.iciba = false
end -- data[#data + 1] = res
-- return
-- if not body.isWord then -- end
-- data.result.youdao = {
-- title = body.query,
-- [data.from == 'en' and 'translation' or 'definition'] = body.translation,
-- }
-- return
-- end
-- local tmp = {
-- title = {
-- word = body.query,
-- phonetic = body.basic.phonetic,
-- },
-- web = body.web,
-- explains = body.basic.explains,
-- -- phrases = body.phrases,
-- -- synonyms = body.synonyms,
-- -- sentenceSample = body.sentenceSample,
-- [data.from == 'en' and 'translation' or 'definition'] = body.translation,
-- }
-- data.result.iciba = tmp
end
require('Trans').curl.get(M.uri, {
query = M.get_content(data),
callback = handle,
})
end end
-- { -- {

View File

@ -1,13 +1,14 @@
---@class Youdao: TransBackend ---@class Youdao: TransOnlineBackend
---@field uri string api uri ---@field uri string api uri
---@field salt string ---@field salt string
---@field app_id string ---@field app_id string
---@field app_passwd string ---@field app_passwd string
---@field disable boolean ---@field disable boolean
local M = { local M = {
uri = 'https://openapi.youdao.com/api', uri = 'https://openapi.youdao.com/api',
salt = tostring(math.random(bit.lshift(1, 15))), salt = tostring(math.random(bit.lshift(1, 15))),
name = 'youdao', name = 'youdao',
method = 'get',
} }
---@class YoudaoQuery ---@class YoudaoQuery
@ -22,7 +23,7 @@ local M = {
---Get content for query ---Get content for query
---@param data TransData ---@param data TransData
---@return YoudaoQuery ---@return YoudaoQuery
function M.get_content(data) function M.get_query(data)
local str = data.str local str = data.str
local app_id = M.app_id local app_id = M.app_id
local salt = M.salt local salt = M.salt
@ -57,28 +58,28 @@ local function check_untracked_field(body)
"phonetic", "phonetic",
'usPhonetic', 'usPhonetic',
"ukPhonetic", "ukPhonetic",
"text", -- text 短语 "text", -- text 短语
"explain", -- String Array 词义解释列表 "explain", -- String Array 词义解释列表
"wordFormats", -- Object Array 单词形式变化列表 "wordFormats", -- Object Array 单词形式变化列表
"name", -- String 形式名称,例如:复数 "name", -- String 形式名称,例如:复数
"phrase", -- String 词组 "phrase", -- String 词组
"meaning", -- String 含义 "meaning", -- String 含义
"synonyms", -- JSONObject 近义词 "synonyms", -- JSONObject 近义词
"pos", -- String 词性 "pos", -- String 词性
"words", -- String Array 近义词列表 "words", -- String Array 近义词列表
"trans", -- String 释义 "trans", -- String 释义
"antonyms", -- ObjectArray 反义词 "antonyms", -- ObjectArray 反义词
"relatedWords", -- JSONArray 相关词 "relatedWords", -- JSONArray 相关词
"wordNet", -- JSONObject 汉语词典网络释义 "wordNet", -- JSONObject 汉语词典网络释义
"phonetic", -- String 发音 "phonetic", -- String 发音
"meanings", -- ObjectArray 释义 "meanings", -- ObjectArray 释义
"meaning", -- String 释义 "meaning", -- String 释义
"example", -- array 示例 "example", -- array 示例
"sentenceSample", -- text 例句 "sentenceSample", -- text 例句
"sentence", -- text 例句 "sentence", -- text 例句
"sentenceBold", -- text 将查询内容加粗的例句 "sentenceBold", -- text 将查询内容加粗的例句
"wfs", -- text 单词形式变化 "wfs", -- text 单词形式变化
"exam_type", -- text 考试类型 "exam_type", -- text 考试类型
} }
for _, f in ipairs(field) do for _, f in ipairs(field) do
if body[f] then if body[f] then
@ -129,58 +130,41 @@ end
---@overload fun(TransData): TransResult ---@overload fun(TransData): TransResult
---Query Using Youdao API ---Query Using Youdao API
---@param data TransData ---@param body table Youdao ouput
function M.query(data) ---@param data TransData Data obj
local handle = function(res) ---@return table|false?
local status, body = pcall(vim.json.decode, res.body) function M.formatter(body, data)
-- vim.print(body) if body.errorCode ~= "0" then return false end
if not status or not body or body.errorCode ~= "0" then check_untracked_field(body)
if not require('Trans').conf.debug then M.debug(body) end
data.result.youdao = false
data[#data + 1] = res
return
end
check_untracked_field(body)
if not body.isWord then
data.result.youdao = {
title = body.query,
[data.from == 'en' and 'translation' or 'definition'] = body.translation,
}
return
end
local tmp = {
title = {
word = body.query,
phonetic = body.basic.phonetic,
},
web = body.web,
explains = body.basic.explains,
-- phrases = body.phrases,
-- synonyms = body.synonyms,
-- sentenceSample = body.sentenceSample,
if not body.isWord then
return {
title = body.query,
[data.from == 'en' and 'translation' or 'definition'] = body.translation, [data.from == 'en' and 'translation' or 'definition'] = body.translation,
} }
data.result.youdao = tmp
end end
require('Trans').curl.get(M.uri, {
query = M.get_content(data), return {
callback = handle, title = {
}) word = body.query,
phonetic = body.basic.phonetic,
},
web = body.web,
explains = body.basic.explains,
-- phrases = body.phrases,
-- synonyms = body.synonyms,
-- sentenceSample = body.sentenceSample,
[data.from == 'en' and 'translation' or 'definition'] = body.translation,
}
end end
---@class TransBackend ---@class TransBackend
---@field youdao Youdao ---@field youdao Youdao
return M return M
-- INFO :Query Result Example
-- { -- {
-- basic = { -- basic = {
-- explains = { "normal", "regular", "normality" }, -- explains = { "normal", "regular", "normality" },

View File

@ -2,11 +2,18 @@ local Trans = require('Trans')
---@class TransBackend ---@class TransBackend
---@field query fun(data: TransData)---@async
---@field no_wait? boolean whether need to wait for the result ---@field no_wait? boolean whether need to wait for the result
---@field all_name string[] @all backend name ---@field all_name string[] @all backend name
---@field name string @backend name ---@field name string @backend name
---@class TransOnlineBackend: TransBackend
---@field uri string @request uri
---@field method 'get' | 'post' @request method
---@field formatter fun(body: table, data: TransData): TransResult|false|nil @formatter
---@field get_query fun(data: TransData): table<string, string> @get query
---@field header? table<string, string> | fun(data: TransData): table<string, string> @request header
---@field debug? fun(body: table?) @debug
local conf = Trans.conf local conf = Trans.conf
--- INFO :Parse online engine keys config file --- INFO :Parse online engine keys config file
@ -21,7 +28,8 @@ if file then
end end
local all_name = {} local all_name = {}
for name, opts in pairs(result) do local backend_order = conf.backend_order or vim.tbl_keys(result)
for name, opts in pairs(backend_order) do
if opts.disable then if opts.disable then
result[name] = nil result[name] = nil
else else
@ -33,7 +41,7 @@ end
---@class Trans ---@class Trans
---@field backend table<string, TransBackend> ---@field backend table<string, TransBackend>
return setmetatable({ return setmetatable({
all_name = Trans.conf.backend_order or all_name, all_name = all_name,
}, { }, {
__index = function(self, name) __index = function(self, name)
---@type TransBackend ---@type TransBackend

View File

@ -9,6 +9,7 @@ local Trans = require('Trans')
---@field mode string @The mode of the str ---@field mode string @The mode of the str
---@field result table<string, TransResult|nil|false> @The result of the translation ---@field result table<string, TransResult|nil|false> @The result of the translation
---@field frontend TransFrontend ---@field frontend TransFrontend
---@field trace table<string, string> debug message
---@field backends table<string, TransBackend> ---@field backends table<string, TransBackend>
local M = {} local M = {}
M.__index = M M.__index = M
@ -28,6 +29,7 @@ function M.new(opts)
str = str, str = str,
mode = mode, mode = mode,
result = {}, result = {},
trace = {},
}, M) }, M)

View File

@ -13,53 +13,32 @@ local function init_opts(opts)
end end
---To Do Online Query
---@param data TransData @data
---@param backend TransOnlineBackend @backend
local function do_query(data, backend) local function do_query(data, backend)
-- TODO : template method for online query -- TODO : template method for online query
local name = backend.name local name = backend.name
local uri = backend.uti local uri = backend.uri
local method = backend.method local method = backend.method
local formatter = backend.formatter local formatter = backend.formatter
local query = backend.get_query(data) local query = backend.get_query(data)
local header = type(backend.header) == "function" and backend.header(data) or backend.header
local header
if backend.header then
if type(backend.header) == "function" then
header = backend.header(data)
else
header = backend.header
end
end
local function handle(output) local function handle(output)
local status, body = pcall(vim.json.decode, output.body) local status, body = pcall(vim.json.decode, output.body)
-- -- vim.print(body) if not status or not body then
if not status or not body or body.errorCode ~= "0" then if not Trans.conf.debug then
if not Trans.conf.debug then backend.debug(body) end backend.debug(body)
data.trace[name] = output
end
data.result[name] = false data.result[name] = false
data[#data + 1] = output
return return
end end
-- check_untracked_field(body)
-- if not body.isWord then -- vim.print(data.result[name])
-- data.result.youdao = { data.result[name] = formatter(body, data)
-- title = body.query,
-- [data.from == 'en' and 'translation' or 'definition'] = body.translation,
-- }
-- return
-- end
-- local tmp = {
-- title = {
-- word = body.query,
-- phonetic = body.basic.phonetic,
-- },
-- web = body.web,
-- explains = body.basic.explains,
-- -- phrases = body.phrases,
-- -- synonyms = body.synonyms,
-- -- sentenceSample = body.sentenceSample,
-- [data.from == 'en' and 'translation' or 'definition'] = body.translation,
-- }
data.result[name] = formatter and formatter(output) or output
end end
Trans.curl[method](uri, { Trans.curl[method](uri, {
@ -70,7 +49,6 @@ local function do_query(data, backend)
-- Hook ? -- Hook ?
end end
---@type table<string, fun(data: TransData): true | nil> ---@type table<string, fun(data: TransData): true | nil>
local strategy = { local strategy = {
fallback = function(data) fallback = function(data)
@ -81,15 +59,13 @@ local strategy = {
local update = data.frontend:wait() local update = data.frontend:wait()
for _, backend in ipairs(data.backends) do for _, backend in ipairs(data.backends) do
do_query(data, backend)
---@cast backend TransBackend ---@cast backend TransBackend
backend.query(data) while result[backend.name] == nil do
local name = backend.name if not update() then break end
while result[name] == nil do
if not update() then return end
end end
if result[name] then return true end if result[backend.name] then return true end
end end
end, end,
--- TODO :More Strategys --- TODO :More Strategys

View File

@ -9,7 +9,6 @@ local M = setmetatable({}, {
local method = self.renderer[name] local method = self.renderer[name]
for _, field in ipairs(order) do for _, field in ipairs(order) do
-- print(field)
method[field](hover, result) method[field](hover, result)
end end
end, end,
@ -62,12 +61,8 @@ default.__index = default
M.renderer = setmetatable({}, { M.renderer = setmetatable({}, {
__index = function(tbl, key) __index = function(tbl, key)
local status, method = pcall(require, 'Trans.frontend.hover.' .. key) local status, method = pcall(require, 'Trans.frontend.hover.' .. key)
if not status then tbl[key] = setmetatable(status and method or {}, default)
print(key) return tbl[key]
return
end
tbl[key] = setmetatable(method, default)
return method
end, end,
}) })