Skip to content

Commit a2c6020

Browse files
committed
Finish 3.1.10
2 parents 737f015 + 7d02c9c commit a2c6020

13 files changed

+221
-58
lines changed

.github/workflows/ci.yml

+7-4
Original file line numberDiff line numberDiff line change
@@ -22,12 +22,11 @@ jobs:
2222
runs-on: ubuntu-latest
2323
env:
2424
CI: true
25-
ALLOW_FAILURES: false ${{ endsWith(matrix.ruby, 'head') }}
25+
ALLOW_FAILURES: ${{ endsWith(matrix.ruby, 'head') || matrix.ruby == 'jruby' }}
2626
strategy:
2727
fail-fast: false
2828
matrix:
2929
ruby:
30-
- 2.4
3130
- 2.5
3231
- 2.6
3332
- 2.7
@@ -44,5 +43,9 @@ jobs:
4443
- name: Install dependencies
4544
run: bundle install --jobs 4 --retry 3
4645
- name: Run tests
47-
run: bundle exec rspec spec || $ALLOW_FAILURES
48-
46+
run: ruby --version; bundle exec rspec spec || $ALLOW_FAILURES
47+
- name: Coveralls GitHub Action
48+
uses: coverallsapp/[email protected]
49+
if: "matrix.ruby == '3.0'"
50+
with:
51+
github-token: ${{ secrets.GITHUB_TOKEN }}

.travis.yml

-16
This file was deleted.

Gemfile

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
source "https://rubygems.org"
22
gem "nokogiri", '~> 1.10'
3-
gem "nokogumbo", platforms: :mri
43

54
gemspec
65
gem 'rdf', git: "https://github.com/ruby-rdf/rdf", branch: "develop"
@@ -38,8 +37,8 @@ group :development do
3837
end
3938

4039
group :development, :test do
41-
gem 'simplecov', platforms: :mri
42-
gem 'coveralls', '~> 0.8', platforms: :mri
40+
gem 'simplecov', '~> 0.21', platforms: :mri
41+
gem 'simplecov-lcov', '~> 0.8', platforms: :mri
4342
gem 'psych', platforms: [:mri, :rbx]
4443
gem 'benchmark-ips'
4544
gem 'rake'

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ JSON::LD parses and serializes [JSON-LD][] into [RDF][] and implements expansion
1414
JSON::LD can now be used to create a _context_ from an RDFS/OWL definition, and optionally include a JSON-LD representation of the ontology itself. This is currently accessed through the `script/gen_context` script.
1515

1616
* If the [jsonlint][] gem is installed, it will be used when validating an input document.
17-
* If available, uses [Nokogiri][] and/or [Nokogumbo][] for parsing HTML, falls back to REXML otherwise.
17+
* If available, uses [Nokogiri][] for parsing HTML, falls back to REXML otherwise.
1818
* Provisional support for [JSON-LD-star][JSON-LD-star].
1919

2020
[Implementation Report](https://ruby-rdf.github.io/json-ld/etc/earl.html)

VERSION

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
3.1.9
1+
3.1.10

lib/json/ld/compact.rb

+1
Original file line numberDiff line numberDiff line change
@@ -297,6 +297,7 @@ def compact(element,
297297
if index_key == '@index'
298298
map_key = expanded_item['@index']
299299
else
300+
index_key = context.expand_iri(index_key, vocab: true)
300301
container_key = context.compact_iri(index_key, vocab: true)
301302
map_key, *others = Array(compacted_item[container_key])
302303
if map_key.is_a?(String)

lib/json/ld/context.rb

+16-12
Original file line numberDiff line numberDiff line change
@@ -253,6 +253,7 @@ def parse(local_context,
253253

254254
local_context = as_array(local_context)
255255

256+
log_depth do
256257
local_context.each do |context|
257258
case context
258259
when nil,false
@@ -266,22 +267,22 @@ def parse(local_context,
266267
"Attempt to clear a context with protected terms"
267268
end
268269
when Context
269-
#log_debug("parse") {"context: #{context.inspect}"}
270+
log_debug("parse") {"context: #{context.inspect}"}
270271
result = result.merge(context)
271272
when IO, StringIO
272-
#log_debug("parse") {"io: #{context}"}
273+
log_debug("parse") {"io: #{context}"}
273274
# Load context document, if it is an open file
274275
begin
275276
ctx = JSON.load(context)
276277
raise JSON::LD::JsonLdError::InvalidRemoteContext, "Context missing @context key" if @options[:validate] && ctx['@context'].nil?
277278
result = result.parse(ctx["@context"] ? ctx["@context"] : {})
278279
rescue JSON::ParserError => e
279-
#log_debug("parse") {"Failed to parse @context from remote document at #{context}: #{e.message}"}
280+
log_info("parse") {"Failed to parse @context from remote document at #{context}: #{e.message}"}
280281
raise JSON::LD::JsonLdError::InvalidRemoteContext, "Failed to parse remote context at #{context}: #{e.message}" if @options[:validate]
281282
self
282283
end
283284
when String, RDF::URI
284-
#log_debug("parse") {"remote: #{context}, base: #{result.context_base || result.base}"}
285+
log_debug("parse") {"remote: #{context}, base: #{result.context_base || result.base}"}
285286

286287
# 3.2.1) Set context to the result of resolving value against the base IRI which is established as specified in section 5.1 Establishing a Base URI of [RFC3986]. Only the basic algorithm in section 5.2 of [RFC3986] is used; neither Syntax-Based Normalization nor Scheme-Based Normalization are performed. Characters additionally allowed in IRI references are treated in the same way that unreserved characters are treated in URI references, per section 6.5 of [RFC3987].
287288
context = RDF::URI(result.context_base || base).join(context)
@@ -296,11 +297,11 @@ def parse(local_context,
296297

297298
cached_context = if PRELOADED[context_canon.to_s]
298299
# If we have a cached context, merge it into the current context (result) and use as the new context
299-
#log_debug("parse") {"=> cached_context: #{context_canon.to_s.inspect}"}
300+
log_debug("parse") {"=> cached_context: #{context_canon.to_s.inspect}"}
300301

301302
# If this is a Proc, then replace the entry with the result of running the Proc
302303
if PRELOADED[context_canon.to_s].respond_to?(:call)
303-
#log_debug("parse") {"=> (call)"}
304+
log_debug("parse") {"=> (call)"}
304305
PRELOADED[context_canon.to_s] = PRELOADED[context_canon.to_s].call
305306
end
306307
PRELOADED[context_canon.to_s]
@@ -320,16 +321,17 @@ def parse(local_context,
320321
ctx = Context.new(unfrozen: true, **options).dup
321322
ctx.context_base = context.to_s
322323
ctx = ctx.parse(remote_doc.document['@context'], remote_contexts: remote_contexts.dup)
324+
ctx.context_base = context.to_s # In case it was altered
323325
ctx.instance_variable_set(:@base, nil)
324326
ctx
325327
end
326328
rescue JsonLdError::LoadingDocumentFailed => e
327-
#log_debug("parse") {"Failed to retrieve @context from remote document at #{context_no_base.context_base.inspect}: #{e.message}"}
329+
log_info("parse") {"Failed to retrieve @context from remote document at #{context_canon.inspect}: #{e.message}"}
328330
raise JsonLdError::LoadingRemoteContextFailed, "#{context}: #{e.message}", e.backtrace
329331
rescue JsonLdError
330332
raise
331333
rescue StandardError => e
332-
#log_debug("parse") {"Failed to retrieve @context from remote document at #{context_no_base.context_base.inspect}: #{e.message}"}
334+
log_info("parse") {"Failed to retrieve @context from remote document at #{context_canon.inspect}: #{e.message}"}
333335
raise JsonLdError::LoadingRemoteContextFailed, "#{context}: #{e.message}", e.backtrace
334336
end
335337
end
@@ -406,6 +408,7 @@ def parse(local_context,
406408
raise JsonLdError::InvalidLocalContext, "must be a URL, JSON object or array of same: #{context.inspect}"
407409
end
408410
end
411+
end
409412
result
410413
end
411414

@@ -475,7 +478,7 @@ def create_term_definition(local_context, term, defined,
475478
remote_contexts: [],
476479
validate_scoped: true)
477480
# Expand a string value, unless it matches a keyword
478-
#log_debug("create_term_definition") {"term = #{term.inspect}"}
481+
log_debug("create_term_definition") {"term = #{term.inspect}"}
479482

480483
# If defined contains the key term, then the associated value must be true, indicating that the term definition has already been created, so return. Otherwise, a cyclical term definition has been detected, which is an error.
481484
case defined[term]
@@ -646,7 +649,7 @@ def create_term_definition(local_context, term, defined,
646649
# Otherwise, term is an absolute IRI. Set the IRI mapping for definition to term
647650
term
648651
end
649-
#log_debug("") {"=> #{definition.id}"}
652+
log_debug("") {"=> #{definition.id}"}
650653
elsif term.include?('/')
651654
# If term is a relative IRI
652655
definition.id = expand_iri(term, vocab: true)
@@ -659,7 +662,7 @@ def create_term_definition(local_context, term, defined,
659662
# Otherwise, active context must have a vocabulary mapping, otherwise an invalid value has been detected, which is an error. Set the IRI mapping for definition to the result of concatenating the value associated with the vocabulary mapping and term.
660663
raise JsonLdError::InvalidIRIMapping, "relative term definition without vocab: #{term} on term #{term.inspect}" unless vocab
661664
definition.id = vocab + term
662-
#log_debug("") {"=> #{definition.id}"}
665+
log_debug("") {"=> #{definition.id}"}
663666
end
664667

665668
@iri_to_term[definition.id] = term if simple_term && definition.id
@@ -699,6 +702,7 @@ def create_term_definition(local_context, term, defined,
699702
when nil then [nil]
700703
else value['@context']
701704
end
705+
log_debug("") {"context: #{definition.context.inspect}"}
702706
rescue JsonLdError => e
703707
raise JsonLdError::InvalidScopedContext, "Term definition for #{term.inspect} contains illegal value for @context: #{e.message}"
704708
end
@@ -1884,7 +1888,7 @@ def remove_base(base, iri)
18841888
@base_and_parents ||= begin
18851889
u = base
18861890
iri_set = u.to_s.end_with?('/') ? [u.to_s] : []
1887-
iri_set << u.to_s while (u = u.parent)
1891+
iri_set << u.to_s while (u != './' && u = u.parent)
18881892
iri_set
18891893
end
18901894
b = base.to_s

lib/json/ld/expand.rb

+11-11
Original file line numberDiff line numberDiff line change
@@ -107,7 +107,7 @@ def expand(input, active_property, context,
107107
Array(input[tk]).sort.each do |term|
108108
term_context = type_scoped_context.term_definitions[term].context if type_scoped_context.term_definitions[term]
109109
unless term_context.nil?
110-
log_debug("expand", depth: log_depth.to_i) {"term_context: #{term_context.inspect}"}
110+
log_debug("expand", depth: log_depth.to_i) {"term_context[#{term}]: #{term_context.inspect}"}
111111
context = context.parse(term_context, base: @options[:base], propagate: false)
112112
end
113113
end
@@ -258,10 +258,10 @@ def expand_object(input, active_property, context, output_object,
258258
expanded_property.to_s.start_with?("_:") &&
259259
context.processingMode('json-ld-1.1')
260260

261-
#log_debug("expand property", depth: log_depth.to_i) {"ap: #{active_property.inspect}, expanded: #{expanded_property.inspect}, value: #{value.inspect}"}
261+
log_debug("expand property", depth: log_depth.to_i) {"ap: #{active_property.inspect}, expanded: #{expanded_property.inspect}, value: #{value.inspect}"}
262262

263263
if expanded_property.nil?
264-
#log_debug(" => ", depth: log_depth.to_i) {"skip nil property"}
264+
log_debug(" => ", depth: log_depth.to_i) {"skip nil property"}
265265
next
266266
end
267267

@@ -341,7 +341,7 @@ def expand_object(input, active_property, context, output_object,
341341
Array(output_object['@included']) + included_result
342342
when '@type'
343343
# If expanded property is @type and value is neither a string nor an array of strings, an invalid type value error has been detected and processing is aborted. Otherwise, set expanded value to the result of using the IRI Expansion algorithm, passing active context, true for vocab, and true for document relative to expand the value or each of its items.
344-
#log_debug("@type", depth: log_depth.to_i) {"value: #{value.inspect}"}
344+
log_debug("@type", depth: log_depth.to_i) {"value: #{value.inspect}"}
345345
e_type = case value
346346
when Array
347347
value.map do |v|
@@ -516,7 +516,7 @@ def expand_object(input, active_property, context, output_object,
516516

517517
# If expanded value contains an @reverse member, i.e., properties that are reversed twice, execute for each of its property and item the following steps:
518518
if value.key?('@reverse')
519-
#log_debug("@reverse", depth: log_depth.to_i) {"double reverse: #{value.inspect}"}
519+
log_debug("@reverse", depth: log_depth.to_i) {"double reverse: #{value.inspect}"}
520520
value['@reverse'].each do |property, item|
521521
# If result does not have a property member, create one and set its value to an empty array.
522522
# Append item to the value of the property member of result.
@@ -566,7 +566,7 @@ def expand_object(input, active_property, context, output_object,
566566
end
567567

568568
# Unless expanded value is null, set the expanded property member of result to expanded value.
569-
#log_debug("expand #{expanded_property}", depth: log_depth.to_i) { expanded_value.inspect}
569+
log_debug("expand #{expanded_property}", depth: log_depth.to_i) { expanded_value.inspect}
570570
output_object[expanded_property] = expanded_value unless expanded_value.nil? && expanded_property == '@value' && input_type != '@json'
571571
next
572572
end
@@ -688,21 +688,21 @@ def expand_object(input, active_property, context, output_object,
688688

689689
# If expanded value is null, ignore key by continuing to the next key from element.
690690
if expanded_value.nil?
691-
#log_debug(" => skip nil value", depth: log_depth.to_i)
691+
log_debug(" => skip nil value", depth: log_depth.to_i)
692692
next
693693
end
694-
#log_debug(depth: log_depth.to_i) {" => #{expanded_value.inspect}"}
694+
log_debug(depth: log_depth.to_i) {" => #{expanded_value.inspect}"}
695695

696696
# If the container mapping associated to key in active context is @list and expanded value is not already a list object, convert expanded value to a list object by first setting it to an array containing only expanded value if it is not already an array, and then by setting it to a JSON object containing the key-value pair @list-expanded value.
697697
if container.first == '@list' && container.length == 1 && !list?(expanded_value)
698-
#log_debug(" => ", depth: log_depth.to_i) { "convert #{expanded_value.inspect} to list"}
698+
log_debug(" => ", depth: log_depth.to_i) { "convert #{expanded_value.inspect} to list"}
699699
expanded_value = {'@list' => as_array(expanded_value)}
700700
end
701-
#log_debug(depth: log_depth.to_i) {" => #{expanded_value.inspect}"}
701+
log_debug(depth: log_depth.to_i) {" => #{expanded_value.inspect}"}
702702

703703
# convert expanded value to @graph if container specifies it
704704
if container.first == '@graph' && container.length == 1
705-
#log_debug(" => ", depth: log_depth.to_i) { "convert #{expanded_value.inspect} to list"}
705+
log_debug(" => ", depth: log_depth.to_i) { "convert #{expanded_value.inspect} to list"}
706706
expanded_value = as_array(expanded_value).map do |v|
707707
{'@graph' => as_array(v)}
708708
end

lib/json/ld/html/nokogiri.rb

+3-4
Original file line numberDiff line numberDiff line change
@@ -136,11 +136,10 @@ def initialize_html_nokogiri(input, options = {})
136136
input
137137
else
138138
begin
139-
require 'nokogumbo' unless defined?(::Nokogumbo)
140139
input = input.read if input.respond_to?(:read)
141-
::Nokogiri::HTML5(input.dup.force_encoding('utf-8'), max_parse_errors: 1000)
142-
rescue LoadError
143-
::Nokogiri::HTML.parse(input, 'utf-8')
140+
::Nokogiri::HTML5(input.force_encoding('utf-8'), max_parse_errors: 1000)
141+
rescue LoadError, NoMethodError
142+
::Nokogiri::HTML.parse(input, base_uri.to_s, 'utf-8')
144143
end
145144
end
146145

spec/compact_spec.rb

+70
Original file line numberDiff line numberDiff line change
@@ -954,6 +954,76 @@
954954
}),
955955
processingMode: 'json-ld-1.1'
956956
},
957+
"issue-514": {
958+
input: %({
959+
"http://example.org/ns/prop": [{
960+
"@id": "http://example.org/ns/bar",
961+
"http://example.org/ns/name": "bar"
962+
}, {
963+
"@id": "http://example.org/ns/foo",
964+
"http://example.org/ns/name": "foo"
965+
}]
966+
}),
967+
context: %({
968+
"@context": {
969+
"ex": "http://example.org/ns/",
970+
"prop": {
971+
"@id": "ex:prop",
972+
"@container": "@index",
973+
"@index": "ex:name"
974+
}
975+
}
976+
}),
977+
output: %({
978+
"@context": {
979+
"ex": "http://example.org/ns/",
980+
"prop": {
981+
"@id": "ex:prop",
982+
"@container": "@index",
983+
"@index": "ex:name"
984+
}
985+
},
986+
"prop": {
987+
"foo": { "@id": "ex:foo"},
988+
"bar": { "@id": "ex:bar"}
989+
}
990+
})
991+
},
992+
"issue-514b": {
993+
input: %({
994+
"http://example.org/ns/prop": [{
995+
"@id": "http://example.org/ns/bar",
996+
"http://example.org/ns/name": "bar"
997+
}, {
998+
"@id": "http://example.org/ns/foo",
999+
"http://example.org/ns/name": "foo"
1000+
}]
1001+
}),
1002+
context: %({
1003+
"@context": {
1004+
"ex": "http://example.org/ns/",
1005+
"prop": {
1006+
"@id": "ex:prop",
1007+
"@container": "@index",
1008+
"@index": "http://example.org/ns/name"
1009+
}
1010+
}
1011+
}),
1012+
output: %({
1013+
"@context": {
1014+
"ex": "http://example.org/ns/",
1015+
"prop": {
1016+
"@id": "ex:prop",
1017+
"@container": "@index",
1018+
"@index": "http://example.org/ns/name"
1019+
}
1020+
},
1021+
"prop": {
1022+
"foo": { "@id": "ex:foo"},
1023+
"bar": { "@id": "ex:bar"}
1024+
}
1025+
})
1026+
},
9571027
}.each_pair do |title, params|
9581028
it(title) {run_compact(params)}
9591029
end

spec/context_spec.rb

+8-1
Original file line numberDiff line numberDiff line change
@@ -181,7 +181,14 @@ def containers
181181
before {JSON::LD::Context.instance_variable_set(:@cache, nil)}
182182
it "retrieves and parses a remote context document" do
183183
expect(JSON::LD::API).to receive(:documentLoader).with("http://example.com/context", anything).and_yield(remote_doc)
184-
subject.parse(ctx)
184+
ec = subject.parse(ctx)
185+
expect(ec.send(:mappings)).to produce({
186+
"xsd" => "http://www.w3.org/2001/XMLSchema#",
187+
"name" => "http://xmlns.com/foaf/0.1/name",
188+
"homepage" => "http://xmlns.com/foaf/0.1/homepage",
189+
"avatar" => "http://xmlns.com/foaf/0.1/avatar",
190+
"integer" => "http://www.w3.org/2001/XMLSchema#integer"
191+
}, logger)
185192
end
186193
end
187194

0 commit comments

Comments
 (0)