2022-09-23 22:00:12 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
class TranslationService::DeepL < TranslationService
|
|
|
|
include JsonLdHelper
|
|
|
|
|
|
|
|
def initialize(plan, api_key)
|
|
|
|
super()
|
|
|
|
|
|
|
|
@plan = plan
|
|
|
|
@api_key = api_key
|
|
|
|
end
|
|
|
|
|
|
|
|
def translate(text, source_language, target_language)
|
2023-03-03 20:06:31 +00:00
|
|
|
form = { text: text, source_lang: source_language&.upcase, target_lang: target_language, tag_handling: 'html' }
|
|
|
|
request(:post, '/v2/translate', form: form) do |res|
|
|
|
|
transform_response(res.body_with_limit)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-03-16 10:07:24 +00:00
|
|
|
def languages
|
|
|
|
source_languages = [nil] + fetch_languages('source')
|
|
|
|
|
|
|
|
# In DeepL, EN and PT are deprecated in favor of EN-GB/EN-US and PT-BR/PT-PT, so
|
|
|
|
# they are supported but not returned by the API.
|
|
|
|
target_languages = %w(en pt) + fetch_languages('target')
|
|
|
|
|
|
|
|
source_languages.index_with { |language| target_languages.without(nil, language) }
|
2023-03-03 20:06:31 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2023-03-16 10:07:24 +00:00
|
|
|
def fetch_languages(type)
|
|
|
|
request(:get, "/v2/languages?type=#{type}") do |res|
|
|
|
|
Oj.load(res.body_with_limit).map { |language| normalize_language(language['language']) }
|
2023-03-03 20:06:31 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-03-16 10:07:24 +00:00
|
|
|
def normalize_language(language)
|
|
|
|
subtags = language.split(/[_-]/)
|
|
|
|
subtags[0].downcase!
|
|
|
|
subtags[1]&.upcase!
|
|
|
|
subtags.join('-')
|
|
|
|
end
|
|
|
|
|
2023-03-03 20:06:31 +00:00
|
|
|
def request(verb, path, **options)
|
|
|
|
req = Request.new(verb, "#{base_url}#{path}", **options)
|
|
|
|
req.add_headers(Authorization: "DeepL-Auth-Key #{@api_key}")
|
|
|
|
req.perform do |res|
|
2022-09-23 22:00:12 +01:00
|
|
|
case res.code
|
|
|
|
when 429
|
|
|
|
raise TooManyRequestsError
|
|
|
|
when 456
|
|
|
|
raise QuotaExceededError
|
|
|
|
when 200...300
|
2023-03-03 20:06:31 +00:00
|
|
|
yield res
|
2022-09-23 22:00:12 +01:00
|
|
|
else
|
|
|
|
raise UnexpectedResponseError
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-03-03 20:06:31 +00:00
|
|
|
def base_url
|
2022-09-23 22:00:12 +01:00
|
|
|
if @plan == 'free'
|
2023-03-03 20:06:31 +00:00
|
|
|
'https://api-free.deepl.com'
|
2022-09-23 22:00:12 +01:00
|
|
|
else
|
2023-03-03 20:06:31 +00:00
|
|
|
'https://api.deepl.com'
|
2022-09-23 22:00:12 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def transform_response(str)
|
|
|
|
json = Oj.load(str, mode: :strict)
|
|
|
|
|
|
|
|
raise UnexpectedResponseError unless json.is_a?(Hash)
|
|
|
|
|
2022-10-24 17:37:57 +01:00
|
|
|
Translation.new(text: json.dig('translations', 0, 'text'), detected_source_language: json.dig('translations', 0, 'detected_source_language')&.downcase, provider: 'DeepL.com')
|
2022-09-23 22:00:12 +01:00
|
|
|
rescue Oj::ParseError
|
|
|
|
raise UnexpectedResponseError
|
|
|
|
end
|
|
|
|
end
|