# frozen_string_literal: true
require "promise"
require "timeout"
module Dhall
module Util
class LazyPromise < Promise
def initialize(&block)
super
@block = block
end
def subscribe(*args)
super
begin
fulfill(@block.call)
rescue => e
reject(e)
end
end
end
class AllOf
def initialize(*validators)
@validators = validators
end
def ===(other)
@validators.all? { |v| v === other }
end
end
class ArrayOf < ValueSemantics::ArrayOf
def initialize(element_validator, min: 0, max: Float::INFINITY)
@min = min
@max = max
super(element_validator)
end
def ===(other)
super && other.length >= @min && other.length <= @max
end
end
class HashOf
def initialize(
key_validator,
element_validator,
min: 0,
max: Float::INFINITY
)
@min = min
@max = max
@key_validator = key_validator
@element_validator = element_validator
end
def ===(other)
Hash === other &&
other.keys.all? { |x| @key_validator === x } &&
other.values.all? { |x| @element_validator === x } &&
other.size >= @min && other.size <= @max
end
end
module ArrayAllTheSame
def self.===(other)
Array === other && other.all? { |x| x == other.first }
end
end
class Not
def initialize(validator)
@validator = validator
end
def ===(other)
!(@validator === other)
end
end
class Deadline
def self.for_timeout(timeout)
if timeout.nil? || timeout.to_f.infinite?
NoDeadline.new
else
new(Time.now + timeout)
end
end
def initialize(deadline)
@deadline = deadline
end
def exceeded?
@deadline < Time.now
end
def timeout
[0.000000000000001, @deadline - Time.now].max
end
def timeout_block(&block)
Timeout.timeout(timeout, TimeoutException, &block)
end
end
class NoDeadline
def exceeded?
false
end
def timeout
nil
end
def timeout_block
yield
end
end
def self.match_results(xs=nil, ys=nil)
Array(xs).each_with_index.map do |r, idx|
yield r, ys[idx]
end
end
def self.match_result_promises(xs=nil, ys=nil)
match_results(yield(Array(xs)), ys) do |promise, promises|
promises.each { |p| p.fulfill(promise) }
end
end
def self.promise_all_hash(hash)
keys, promises = hash.to_a.transpose
return Promise.resolve(hash) unless keys
Promise.all(promises).then do |values|
Hash[Util.match_results(keys, values) do |k, v|
[k, v]
end]
end
end
def self.psych_coder_for(tag, v)
c = Psych::Coder.new(tag)
case v
when Hash
c.map = v
when Array
c.seq = v
else
c.scalar = v
end
end
def self.psych_coder_from(tag, o)
coder = Psych::Coder.new(tag)
if o.respond_to?(:encode_with)
o.encode_with(coder)
else
o.instance_variables.each do |ivar|
coder[ivar.to_s[1..-1]] = o.instance_variable_get(ivar)
end
end
coder
end
def self.transform_keys(hash_or_not)
return hash_or_not unless hash_or_not.is_a?(Hash)
Hash[hash_or_not.map { |k, v| [(yield k), v] }]
end
def self.utf8_if_possible(str)
utf8 = str.dup.force_encoding(Encoding::UTF_8)
utf8.valid_encoding? ? utf8 : str
end
def self.text_or_binary(str)
unless str.valid_encoding?
raise ArgumentError, "invalid byte sequence in #{str.encoding}"
end
if str.encoding == Encoding::BINARY
return str if str =~ /(?!\s)[[:cntrl:]]/
utf8_if_possible(str)
else
str.encode(Encoding::UTF_8)
end
end
def self.longest_common_prefix(a, b)
a.zip(b).take_while { |(x, y)| x == y }.map(&:first)
end
def self.indent_size(str)
if str.end_with?("\n")
0
else
str
.scan(/^[ \t]*(?=[^ \t\n]|\Z)/)
.map(&:chars)
.reduce(&method(:longest_common_prefix))&.length.to_i
end
end
def self.path_components_to_uri(*components)
URI("/#{components.map(&method(:uri_escape)).join("/")}")
end
def self.uri_escape(s)
::URI.encode_www_form_component(s).gsub("+", "%20")
end
def self.net_http_req_with_timeout(uri, req, timeout:)
Net::HTTP.start(
uri.hostname,
uri.port,
use_ssl: uri.scheme == "https",
open_timeout: timeout,
ssl_timeout: timeout,
read_timeout: timeout,
write_timeout: timeout
) { |http| http.request(req) }
end
end
end