From aee97aca2b89e749d5dcc48c421fdb79be7b70ed Mon Sep 17 00:00:00 2001 From: Ronald Tse Date: Wed, 7 Sep 2022 11:23:06 +0800 Subject: [PATCH] fix: refactor datauri methods, fixes #20 --- lib/utils/image.rb | 65 +++++++++++++++++++++++++++++++--------------- spec/img_spec.rb | 11 -------- 2 files changed, 44 insertions(+), 32 deletions(-) diff --git a/lib/utils/image.rb b/lib/utils/image.rb index 45e5cf8..91f2e5c 100644 --- a/lib/utils/image.rb +++ b/lib/utils/image.rb @@ -1,7 +1,5 @@ -require "asciidoctor" require "tempfile" require "marcel" -require "mime/types" require "base64" module Metanorma @@ -126,35 +124,60 @@ def svg_update_ids_css(svg, ids, idx) # sources/plantuml/plantuml20200524-90467-1iqek5i.png # already includes localdir - def datauri(uri, localdirectory = ".") - return uri if /^data:/.match?(uri) + def datauri(uri, local_dir = ".") - path = datauri_path(uri, localdirectory) - return path unless File.exist?(path) + # Return the data URI if it already is a data URI + return uri if is_datauri?(uri) - types = MIME::Types.type_for(path) - type = types ? types.first.to_s : 'text/plain; charset="utf-8"' - bin = File.open(path, "rb", &:read) + # Return the URL if it is a URL + return uri if is_url?(uri) + + local_path = uri + relative_path = File.join(local_dir, uri) + + # Check whether just the local path or the other specified relative path + # works. + path = [ local_path, relative_path ].detect do |p| + File.exist?(p) ? p : nil + end + + unless path && File.exist?(path) + warn "Image specified at `#{uri}` does not exist." + # Return original provided location + return uri + end + + encode_datauri(path) + end + + def encode_datauri(path) + return nil unless File.exist?(path) + + type = Marcel::MimeType.for(Pathname.new(path)) || + 'text/plain; charset="utf-8"' + + bin = File.open(path, "rb").read data = Base64.strict_encode64(bin) "data:#{type};base64,#{data}" + rescue + warn "Data-URI encoding of `#{path}` failed." + nil end - def datauri_path(uri, localdirectory) - path = if %r{^([A-Z]:)?/}.match?(uri) then uri - else - File.exist?(uri) ? uri : File.join(localdirectory, uri) - end - unless File.exist?(path) - warn "image at #{path} not found" - return uri - end - path + def is_datauri?(uri) + /^data:/.match?(uri) + end + + def is_url?(url) + %r{^([A-Z]:)?/}.match?(url) end # FIXME: This method should ONLY return 1 type, remove Array wrapper def datauri2mime(uri) - %r{^data:image/(?[^;]+);base64,(?.+)$} =~ uri - [Marcel::MimeType.for(Base64.strict_decode64(imgdata))] + %r{^data:(?[^;]+);base64,(?.+)$} =~ uri + return nil unless mimetype && mimedata + + [Marcel::MimeType.for(Base64.strict_decode64(mimedata), declared_type: mimetype)] end end end diff --git a/spec/img_spec.rb b/spec/img_spec.rb index 81582cd..e877a47 100644 --- a/spec/img_spec.rb +++ b/spec/img_spec.rb @@ -276,17 +276,6 @@ OUTPUT end - def datauri(uri, localdir = "") - return uri if /^data:/.match?(uri) - - path = File.join(localdir, uri) - types = MIME::Types.type_for(path) - type = types ? types.first.to_s : 'text/plain; charset="utf-8"' - bin = File.open(path, "rb", &:read) - data = Base64.strict_encode64(bin) - "data:#{type};base64,#{data}" - end - it "generates data uris" do expect(Metanorma::Utils.datauri("data:xyz")).to eq "data:xyz" expect(Metanorma::Utils.datauri("spec/fixtures/rice_image1.png")).to be_equivalent_to ""