Skip to content

Commit

Permalink
Add Language Detection endpoint
Browse files Browse the repository at this point in the history
Close #247

Test
  • Loading branch information
tagliala committed Sep 27, 2024
1 parent 8dce844 commit 3b5afc4
Show file tree
Hide file tree
Showing 11 changed files with 165 additions and 6 deletions.
3 changes: 2 additions & 1 deletion .rubocop_todo.yml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

24 changes: 24 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -343,6 +343,30 @@ Response:

... PDF document body ...

### Detect language

This is a foreground document language detection request. The detected language
will be returned as the response body.

POST /convert

Params *(suggest using `multipart/form-data`)*:

* `file` - the file to detect
* `action` - `detect_language`

#### Example:

POST /convert
file=... foo.docx ...
action=detect_language

Response:

Content-Type: text/plain

en

## Callbacks

When a document conversion is completed, an attempt will be made to POST a
Expand Down
1 change: 1 addition & 0 deletions config/app.yml
Original file line number Diff line number Diff line change
Expand Up @@ -33,3 +33,4 @@ wkhtmltopdf_path: <%= ENV['WKHTMLTOPDF_PATH'] %>
# Other settings
tika_config_directory: <%= ENV['TIKA_CONFIG_DIRECTORY'] %>
wkhtmltopdf_params: '-d 100 --encoding UTF-8'
tesseract_available_languages: <%= ENV['TESSERACT_AVAILABLE_LANGUAGES'] %>
22 changes: 22 additions & 0 deletions lib/app.rb
Original file line number Diff line number Diff line change
Expand Up @@ -192,6 +192,28 @@ class App < Sinatra::Base
respond_with_error e
end

#
# Detect document language
#
# POST params:
# file - the file to detect language
post '/detect-language' do
unless params[:file]
return respond 400, "missing file parameter"
end

unless params[:file].respond_to?(:fetch) and params[:file].fetch(:tempfile, nil).respond_to?(:read)
return respond 400, "invalid file parameter"
end

body = params[:file][:tempfile].read
content = Converter.new(logger: @logger).convert_file('detect-language', body)
content_type content.mime_type
content
rescue StandardError => e
respond_with_error e
end

# Legacy method to convert files
# Brought over from Heathen
#
Expand Down
3 changes: 3 additions & 0 deletions lib/config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,8 @@ class C_
attr_accessor :tika_config_directory
# @return [String] Params for wkhtmltopdf
attr_accessor :wkhtmltopdf_params
# @return [Array<String>] Languages available to Tesseract for OCR. Defaults to `["eng"]`
attr_accessor :tesseract_available_languages

def self.config_file_path
Pathname.new File.expand_path('../config/app.yml', __dir__)
Expand All @@ -70,6 +72,7 @@ def self.config

c.tika_config_directory = yaml['tika_config_directory'] || '../tmp/tika'
c.wkhtmltopdf_params = yaml['wkhtmltopdf_params'] || ''
c.tesseract_available_languages = yaml['tesseract_available_languages'].to_s.split(',') || %w[eng]

c
end
Expand Down
18 changes: 18 additions & 0 deletions lib/heathen/processor_methods/detect_language.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# frozen_string_literal: true

module Heathen
class Processor
def detect_language
executioner.execute(
Colore::C_.tika_path,
"--config=#{Colore::C_.tika_config}",
'--language',
job.content_file,
binary: true
)
raise ConversionFailed.new if executioner.last_exit_status != 0

job.content = executioner.stdout
end
end
end
4 changes: 4 additions & 0 deletions lib/heathen/task.rb
Original file line number Diff line number Diff line change
Expand Up @@ -103,3 +103,7 @@ def task_key(action, mime_type)
Heathen::Task.register 'doc', '.*' do
perform_task 'msoffice'
end

Heathen::Task.register 'detect_language', '.*' do
detect_language
end
22 changes: 17 additions & 5 deletions lib/tika_config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ module TikaConfig
<parser class="org.apache.tika.parser.DefaultParser"></parser>
<parser class="org.apache.tika.parser.ocr.TesseractOCRParser">
<params>
<param name="language" type="string">%<language_alpha3>s</param>
<param name="language" type="string">%<alpha3_languages>s</param>
</params>
</parser>
</parsers>
Expand All @@ -34,14 +34,18 @@ def tika_config_path
Pathname.new File.expand_path(Colore::C_.tika_config_directory, __dir__)
end

def path_for!(language_alpha3)
file = tika_config_path.join('ocr', VERSION, "tika.#{language_alpha3}.xml")
def path_for!(alpha3_languages, filename:)
file = tika_config_path.join('ocr', VERSION, "tika.#{filename}.xml")
return file if file.file?

FileUtils.mkdir_p(tika_config_path.join('ocr', VERSION))
file.write format(TEMPLATE, language_alpha3: language_alpha3)
file.write format(TEMPLATE, alpha3_languages: alpha3_languages.join('+'))
file
end

def path_for_language!(language_alpha3)
path_for!([language_alpha3], filename: language_alpha3)
end
end

# Returns the file path of the Tika configuration for performing OCR
Expand All @@ -55,7 +59,15 @@ def path_for!(language_alpha3)
def self.path_for(language)
language_alpha3 = Colore::Utils.language_alpha3(language) || DEFAULT_LANGUAGE

path_for!(language_alpha3)
path_for_language!(language_alpha3)
end

# Returns the file path of the Tika configuration for performing language
# detection.
#
# @return [Pathname] The path to the Tika configuration file for language detection
def self.path_for_language_detection
path_for!(Colore::C_.tesseract_available_languages, filename: 'language_detection')
end
end
end
37 changes: 37 additions & 0 deletions spec/heathen/processor_methods/detect_language_spec.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe Heathen::Processor do
let(:content) { fixture('heathen/quickfox.ar.jpg').read }
let(:job) { Heathen::Job.new 'foo', content }
let(:processor) { described_class.new job: job, logger: spec_logger }

after do
processor.clean_up
end

describe '#detect_language' do
before do
processor.detect_language
end

context 'with English documents' do
let(:content) { fixture('heathen/quickfox.jpg').read }

it 'detects English input file language' do
expect(job.content).to eq 'en'
expect(job.content.mime_type).to eq 'text/plain; charset=us-ascii'
end
end

context 'with Arabic documents' do
let(:content) { fixture('heathen/quickfox.ar.jpg').read }

it 'detects Arabic input file language' do
expect(job.content).to eq 'ar'
expect(job.content.mime_type).to eq 'text/plain; charset=us-ascii'
end
end
end
end
8 changes: 8 additions & 0 deletions spec/integration/standard_tasks_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,14 @@
end
end

describe 'detect_language' do
it 'runs' do
content = fixture('heathen/quickfox.jpg').read
new_content = converter.convert 'detect_language', content
expect(new_content.mime_type).to eq 'text/plain; charset=us-ascii'
end
end

describe 'ocr_text' do
it 'converts jpeg' do
content = fixture('heathen/quickfox.jpg').read
Expand Down
29 changes: 29 additions & 0 deletions spec/lib/tika_config_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -56,4 +56,33 @@
end
end
end

describe '.path_for_language_detection' do
subject(:path_for_language_detection) { described_class.path_for_language_detection }

before do
allow(Colore::C_.config).to receive(:tesseract_available_languages).and_return(%w[eng fra])
end

it 'returns the correct configuration file path' do
expect(path_for_language_detection).to eq tika_test_config_path.join('ocr', described_class::VERSION, 'tika.language_detection.xml')
end

it 'includes all the available languages' do
expect(path_for_language_detection.read).to include('eng+fra')
end

context 'when the configuration file is already present' do
before do
allow(FileUtils).to receive(:mkdir_p)
.with(tika_test_config_path.join('ocr', described_class::VERSION))
.and_call_original
end

it 'does not overwrite it' do
2.times { described_class.path_for_language_detection }
expect(FileUtils).to have_received(:mkdir_p).once
end
end
end
end

0 comments on commit 3b5afc4

Please sign in to comment.