linked_vocabs 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 8afd0f6c1a9ef5c1b998260ea65516f997d8450e
4
+ data.tar.gz: 16715bdd289fed07c8a13b0ad1c0efeef41167d0
5
+ SHA512:
6
+ metadata.gz: 7be672326fb2ee98c5d39407f26d461b008c798e8ba6a6f5aa8318cc3132341f7fb7b4002f476c1113158587ede9a283ef1fb1909fa974c0e3f91ec3b376debc
7
+ data.tar.gz: 8c26071cf7e3bfbe01bcfd5168c4d6706d32a18c4e3ead87e882f1c69bc631676c210c3d47b13d573a07c7061c9be6e4f627d8d2f169b931f5199d894c1e9979
data/.gitignore ADDED
@@ -0,0 +1,18 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ .idea
7
+ Gemfile.lock
8
+ InstalledFiles
9
+ _yardoc
10
+ coverage
11
+ doc/
12
+ lib/bundler/man
13
+ pkg
14
+ rdoc
15
+ spec/reports
16
+ test/tmp
17
+ test/version_tmp
18
+ tmp
data/Gemfile ADDED
@@ -0,0 +1,6 @@
1
+ source 'https://rubygems.org'
2
+
3
+ gem 'pry'
4
+ # Specify your gem's dependencies in hybag.gemspec
5
+ gemspec
6
+
data/LICENSE.txt ADDED
@@ -0,0 +1,191 @@
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction, and
10
+ distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by the copyright
13
+ owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all other entities
16
+ that control, are controlled by, or are under common control with that entity.
17
+ For the purposes of this definition, "control" means (i) the power, direct or
18
+ indirect, to cause the direction or management of such entity, whether by
19
+ contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the
20
+ outstanding shares, or (iii) beneficial ownership of such entity.
21
+
22
+ "You" (or "Your") shall mean an individual or Legal Entity exercising
23
+ permissions granted by this License.
24
+
25
+ "Source" form shall mean the preferred form for making modifications, including
26
+ but not limited to software source code, documentation source, and configuration
27
+ files.
28
+
29
+ "Object" form shall mean any form resulting from mechanical transformation or
30
+ translation of a Source form, including but not limited to compiled object code,
31
+ generated documentation, and conversions to other media types.
32
+
33
+ "Work" shall mean the work of authorship, whether in Source or Object form, made
34
+ available under the License, as indicated by a copyright notice that is included
35
+ in or attached to the work (an example is provided in the Appendix below).
36
+
37
+ "Derivative Works" shall mean any work, whether in Source or Object form, that
38
+ is based on (or derived from) the Work and for which the editorial revisions,
39
+ annotations, elaborations, or other modifications represent, as a whole, an
40
+ original work of authorship. For the purposes of this License, Derivative Works
41
+ shall not include works that remain separable from, or merely link (or bind by
42
+ name) to the interfaces of, the Work and Derivative Works thereof.
43
+
44
+ "Contribution" shall mean any work of authorship, including the original version
45
+ of the Work and any modifications or additions to that Work or Derivative Works
46
+ thereof, that is intentionally submitted to Licensor for inclusion in the Work
47
+ by the copyright owner or by an individual or Legal Entity authorized to submit
48
+ on behalf of the copyright owner. For the purposes of this definition,
49
+ "submitted" means any form of electronic, verbal, or written communication sent
50
+ to the Licensor or its representatives, including but not limited to
51
+ communication on electronic mailing lists, source code control systems, and
52
+ issue tracking systems that are managed by, or on behalf of, the Licensor for
53
+ the purpose of discussing and improving the Work, but excluding communication
54
+ that is conspicuously marked or otherwise designated in writing by the copyright
55
+ owner as "Not a Contribution."
56
+
57
+ "Contributor" shall mean Licensor and any individual or Legal Entity on behalf
58
+ of whom a Contribution has been received by Licensor and subsequently
59
+ incorporated within the Work.
60
+
61
+ 2. Grant of Copyright License.
62
+
63
+ Subject to the terms and conditions of this License, each Contributor hereby
64
+ grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
65
+ irrevocable copyright license to reproduce, prepare Derivative Works of,
66
+ publicly display, publicly perform, sublicense, and distribute the Work and such
67
+ Derivative Works in Source or Object form.
68
+
69
+ 3. Grant of Patent License.
70
+
71
+ Subject to the terms and conditions of this License, each Contributor hereby
72
+ grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
73
+ irrevocable (except as stated in this section) patent license to make, have
74
+ made, use, offer to sell, sell, import, and otherwise transfer the Work, where
75
+ such license applies only to those patent claims licensable by such Contributor
76
+ that are necessarily infringed by their Contribution(s) alone or by combination
77
+ of their Contribution(s) with the Work to which such Contribution(s) was
78
+ submitted. If You institute patent litigation against any entity (including a
79
+ cross-claim or counterclaim in a lawsuit) alleging that the Work or a
80
+ Contribution incorporated within the Work constitutes direct or contributory
81
+ patent infringement, then any patent licenses granted to You under this License
82
+ for that Work shall terminate as of the date such litigation is filed.
83
+
84
+ 4. Redistribution.
85
+
86
+ You may reproduce and distribute copies of the Work or Derivative Works thereof
87
+ in any medium, with or without modifications, and in Source or Object form,
88
+ provided that You meet the following conditions:
89
+
90
+ You must give any other recipients of the Work or Derivative Works a copy of
91
+ this License; and
92
+ You must cause any modified files to carry prominent notices stating that You
93
+ changed the files; and
94
+ You must retain, in the Source form of any Derivative Works that You distribute,
95
+ all copyright, patent, trademark, and attribution notices from the Source form
96
+ of the Work, excluding those notices that do not pertain to any part of the
97
+ Derivative Works; and
98
+ If the Work includes a "NOTICE" text file as part of its distribution, then any
99
+ Derivative Works that You distribute must include a readable copy of the
100
+ attribution notices contained within such NOTICE file, excluding those notices
101
+ that do not pertain to any part of the Derivative Works, in at least one of the
102
+ following places: within a NOTICE text file distributed as part of the
103
+ Derivative Works; within the Source form or documentation, if provided along
104
+ with the Derivative Works; or, within a display generated by the Derivative
105
+ Works, if and wherever such third-party notices normally appear. The contents of
106
+ the NOTICE file are for informational purposes only and do not modify the
107
+ License. You may add Your own attribution notices within Derivative Works that
108
+ You distribute, alongside or as an addendum to the NOTICE text from the Work,
109
+ provided that such additional attribution notices cannot be construed as
110
+ modifying the License.
111
+ You may add Your own copyright statement to Your modifications and may provide
112
+ additional or different license terms and conditions for use, reproduction, or
113
+ distribution of Your modifications, or for any such Derivative Works as a whole,
114
+ provided Your use, reproduction, and distribution of the Work otherwise complies
115
+ with the conditions stated in this License.
116
+
117
+ 5. Submission of Contributions.
118
+
119
+ Unless You explicitly state otherwise, any Contribution intentionally submitted
120
+ for inclusion in the Work by You to the Licensor shall be under the terms and
121
+ conditions of this License, without any additional terms or conditions.
122
+ Notwithstanding the above, nothing herein shall supersede or modify the terms of
123
+ any separate license agreement you may have executed with Licensor regarding
124
+ such Contributions.
125
+
126
+ 6. Trademarks.
127
+
128
+ This License does not grant permission to use the trade names, trademarks,
129
+ service marks, or product names of the Licensor, except as required for
130
+ reasonable and customary use in describing the origin of the Work and
131
+ reproducing the content of the NOTICE file.
132
+
133
+ 7. Disclaimer of Warranty.
134
+
135
+ Unless required by applicable law or agreed to in writing, Licensor provides the
136
+ Work (and each Contributor provides its Contributions) on an "AS IS" BASIS,
137
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
138
+ including, without limitation, any warranties or conditions of TITLE,
139
+ NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are
140
+ solely responsible for determining the appropriateness of using or
141
+ redistributing the Work and assume any risks associated with Your exercise of
142
+ permissions under this License.
143
+
144
+ 8. Limitation of Liability.
145
+
146
+ In no event and under no legal theory, whether in tort (including negligence),
147
+ contract, or otherwise, unless required by applicable law (such as deliberate
148
+ and grossly negligent acts) or agreed to in writing, shall any Contributor be
149
+ liable to You for damages, including any direct, indirect, special, incidental,
150
+ or consequential damages of any character arising as a result of this License or
151
+ out of the use or inability to use the Work (including but not limited to
152
+ damages for loss of goodwill, work stoppage, computer failure or malfunction, or
153
+ any and all other commercial damages or losses), even if such Contributor has
154
+ been advised of the possibility of such damages.
155
+
156
+ 9. Accepting Warranty or Additional Liability.
157
+
158
+ While redistributing the Work or Derivative Works thereof, You may choose to
159
+ offer, and charge a fee for, acceptance of support, warranty, indemnity, or
160
+ other liability obligations and/or rights consistent with this License. However,
161
+ in accepting such obligations, You may act only on Your own behalf and on Your
162
+ sole responsibility, not on behalf of any other Contributor, and only if You
163
+ agree to indemnify, defend, and hold each Contributor harmless for any liability
164
+ incurred by, or claims asserted against, such Contributor by reason of your
165
+ accepting any such warranty or additional liability.
166
+
167
+ END OF TERMS AND CONDITIONS
168
+
169
+ APPENDIX: How to apply the Apache License to your work
170
+
171
+ To apply the Apache License to your work, attach the following boilerplate
172
+ notice, with the fields enclosed by brackets "[]" replaced with your own
173
+ identifying information. (Don't include the brackets!) The text should be
174
+ enclosed in the appropriate comment syntax for the file format. We also
175
+ recommend that a file or class name and description of purpose be included on
176
+ the same "printed page" as the copyright notice for easier identification within
177
+ third-party archives.
178
+
179
+ Copyright [yyyy] [name of copyright owner]
180
+
181
+ Licensed under the Apache License, Version 2.0 (the "License");
182
+ you may not use this file except in compliance with the License.
183
+ You may obtain a copy of the License at
184
+
185
+ http://www.apache.org/licenses/LICENSE-2.0
186
+
187
+ Unless required by applicable law or agreed to in writing, software
188
+ distributed under the License is distributed on an "AS IS" BASIS,
189
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
190
+ See the License for the specific language governing permissions and
191
+ limitations under the License.
data/README.md ADDED
@@ -0,0 +1,27 @@
1
+ # Linked Vocabularies
2
+
3
+ Linked Data Controlled Vocabularies for ActiveFedora::Rdf
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'linked_vocabs'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install linked_vocabs
18
+
19
+ ## Usage
20
+
21
+ ## Contributing
22
+
23
+ 1. Fork it
24
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
25
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
26
+ 4. Push to the branch (`git push origin my-new-feature`)
27
+ 5. Create new Pull Request
data/Rakefile ADDED
@@ -0,0 +1,17 @@
1
+ #!/usr/bin/env rake
2
+
3
+ require 'linked_vocabs'
4
+
5
+ desc "Generate Vocabularies"
6
+
7
+ task :gen_vocabs do
8
+ LinkedVocabs.vocabularies.each_key do |name|
9
+ puts "Generating vocabulary at lib/rdf/#{name}.rb"
10
+ begin
11
+ LinkedVocabs.load_vocabulary(name)
12
+ rescue
13
+ puts "Failed to load #{name}: #{$!.message}"
14
+ puts $!.backtrace
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,75 @@
1
+ require 'active_triples'
2
+ require 'active_model'
3
+ require 'rdf/cli/vocab-loader'
4
+
5
+ Dir['./lib/rdf/*.rb'].each { |f| require f }
6
+
7
+ require 'linked_vocabs/version'
8
+ require 'linked_vocabs/validators'
9
+ require 'linked_vocabs/controlled'
10
+
11
+ module LinkedVocabs
12
+
13
+ def vocabularies
14
+ @vocabularies ||= {
15
+ :dcmitype => { :prefix => 'http://purl.org/dc/dcmitype/', :source => 'http://dublincore.org/2012/06/14/dctype.rdf' },
16
+ :geonames => { :prefix => 'http://sws.geonames.org/', :strict => false, :fetch => false },
17
+ :lcsh => { :prefix => 'http://id.loc.gov/authorities/subjects/', :strict => false, :fetch => false }
18
+ }
19
+ end
20
+ module_function :vocabularies
21
+
22
+ def add_vocabulary(name, prefix, args = {})
23
+ name = name.to_sym
24
+ source = args.delete :source
25
+ strict = args.delete :strict
26
+ fetch = args.delete :fetch
27
+ raise "Unexpected arguments #{args.keys}. Accepted parameters are :source, :strict, and :fetch." unless args.empty?
28
+ vocabularies[name] = {
29
+ :prefix => prefix.to_s,
30
+ :strict => strict,
31
+ :fetch => fetch
32
+ }
33
+ vocabularies[name][:source] = source if source
34
+ vocabularies[name][:strict] = false if vocabularies[name][:strict].nil? or !source
35
+ vocabularies[name][:fetch] = false if vocabularies[name][:fetch].nil? or !source
36
+ end
37
+ module_function :add_vocabulary
38
+
39
+ # Based closely on https://github.com/ruby-rdf/rdf/blob/develop/Rakefile
40
+ def load_vocabulary(name, path = 'lib/rdf/')
41
+ raise "Unregistered vocabulary #{name}" unless vocabularies.has_key? name
42
+ v = vocabularies[name]
43
+ out = StringIO.new
44
+ class_name = name.to_s.upcase
45
+ if v.fetch(:fetch, true)
46
+ loader = RDF::VocabularyLoader.new(class_name)
47
+ loader.uri = v[:prefix]
48
+ loader.source = v[:source] if v[:source]
49
+ loader.extra = v[:extra] if v[:extra]
50
+ loader.strict = v.fetch(:strict, true)
51
+ loader.output = out
52
+ loader.run
53
+ else
54
+ out.print %(# -*- encoding: utf-8 -*-
55
+ # This file generated automatically using vocab-fetch from #{v.fetch(:source, v[:prefix])}
56
+ require 'rdf'
57
+ module RDF
58
+ class #{class_name} < RDF::#{"Strict" if v.fetch(:strict, true)}Vocabulary("#{v[:prefix]}")
59
+ # terms not fetched by vocab-fetch
60
+ end
61
+ end
62
+ ).gsub(/^ /, '')
63
+ end
64
+ out.rewind
65
+ File.open(File.join(path, "#{name}.rb"), "w") {|f| f.write out.read}
66
+ end
67
+ module_function :load_vocabulary
68
+
69
+ def load_all
70
+ vocabularies.each_key do |name|
71
+ load_vocabulary(name)
72
+ end
73
+ end
74
+ module_function :load_all
75
+ end
@@ -0,0 +1,199 @@
1
+ require 'linkeddata'
2
+ require 'active_model'
3
+ require 'active_support/core_ext/module/delegation'
4
+
5
+ module LinkedVocabs
6
+ ##
7
+ # Adds add support for controlled vocabularies and
8
+ # QuestioningAuthority to RdfResource classes.
9
+ # @TODO: introduce graph context for provenance
10
+ module Controlled
11
+
12
+ def self.included(klass)
13
+ klass.extend ClassMethods
14
+ klass.property :hiddenLabel, :predicate => RDF::SKOS.hiddenLabel
15
+ klass.validates_with LinkedVocabs::Validators::AuthorityValidator
16
+ end
17
+
18
+ def qa_interface
19
+ self.class.qa_interface
20
+ end
21
+ delegate :search, :get_full_record, :response, :results, :to => :qa_interface
22
+
23
+
24
+ # Override set_subject! to find terms when (and only when) they
25
+ # exist in the vocabulary
26
+ def set_subject!(uri_or_str)
27
+ vocab_matches = []
28
+ begin
29
+ uri = get_uri(uri_or_str)
30
+ uri_or_str = uri
31
+ rescue RuntimeError, NoMethodError
32
+ end
33
+
34
+ self.class.vocabularies.each do |vocab, config|
35
+ if uri_or_str.start_with? config[:prefix]
36
+ # @TODO: is it good to need a full URI for a non-strict vocab?
37
+ return super if config[:strict] == false
38
+ uri_stub = uri_or_str.to_s.gsub(config[:prefix], '')
39
+ return super(config[:class].send(uri_stub)) if config[:class].respond_to? uri_stub
40
+ else
41
+ # this only matches if the term is explictly defined
42
+ # @TODO: what about the possibility of terms like "entries" or
43
+ # "map" which are methods but not defined properties? does
44
+ # this need to be patched in RDF::Vocabulary or am I missing
45
+ # something?
46
+ vocab_matches << config[:class].send(uri_or_str) if config[:class].respond_to? uri_or_str
47
+ end
48
+ end
49
+ return super if vocab_matches.empty?
50
+ uri_or_str = vocab_matches.first
51
+ return super if self.class.uses_vocab_prefix?(uri_or_str) and not uri_or_str.kind_of? RDF::Node
52
+ end
53
+
54
+ def in_vocab?
55
+ return false unless self.class.uses_vocab_prefix?(rdf_subject.to_s)
56
+ self.class.vocabularies.each do |vocab, config|
57
+ return false if rdf_subject == config[:prefix]
58
+ return false if config[:class].strict? and not config[:class].respond_to? rdf_subject.to_s.gsub(config[:prefix], '').to_sym
59
+ end
60
+ true
61
+ end
62
+
63
+ def rdf_label
64
+ labels = Array(self.class.rdf_label)
65
+ labels += default_labels
66
+ labels.each do |label|
67
+ values = get_values(label) if values.blank?
68
+ return values unless values.empty?
69
+ end
70
+ node? ? [] : [rdf_subject.to_s]
71
+ end
72
+
73
+ ##
74
+ # Class methods for adding and using controlled vocabularies
75
+ module ClassMethods
76
+ def use_vocabulary(name, opts={})
77
+ raise ControlledVocabularyError, "Vocabulary undefined: #{name.to_s.upcase}" unless LinkedVocabs.vocabularies.include? name
78
+ opts[:class] = name_to_class(name) unless opts.include? :class
79
+ opts.merge! LinkedVocabs.vocabularies[name.to_sym]
80
+ vocabularies[name] = opts
81
+ end
82
+
83
+ def vocabularies
84
+ @vocabularies ||= {}.with_indifferent_access
85
+ end
86
+
87
+ ##
88
+ # @return [Array<RDF::URI>] terms allowable by the registered StrictVocabularies
89
+ #
90
+ # Note: this does not necessarily list *all the term* allowable
91
+ # by the class. Non-strict RDF::Vocabularies are not included in
92
+ # this method's output.
93
+ def list_terms
94
+ terms = []
95
+ vocabularies.each do |vocab, config|
96
+ next unless config[:class].respond_to? :properties
97
+ terms += config[:class].properties.select { |s| s.start_with? config[:class].to_s }
98
+ end
99
+ terms
100
+ end
101
+
102
+ ##
103
+ # Gets data for all vocabularies used and loads it into the
104
+ # configured repository. After running this new (and reloaded)
105
+ # RdfResource objects of this class will have data from their
106
+ # source web document.
107
+ def load_vocabularies
108
+ vocabularies.each do |name, config|
109
+ load_vocab(name)
110
+ end
111
+ end
112
+
113
+ def uses_vocab_prefix?(str)
114
+ vocabularies.each do |vocab, config|
115
+ return true if str.start_with? config[:prefix]
116
+ end
117
+ false
118
+ end
119
+
120
+ def qa_interface
121
+ @qa_interface ||= QaRDF.new(self)
122
+ end
123
+
124
+ private
125
+
126
+ def name_to_class(name)
127
+ "RDF::#{name.upcase.to_s}".constantize
128
+ end
129
+
130
+ def load_vocab(name)
131
+ return nil unless LinkedVocabs.vocabularies[name.to_sym].include? :source
132
+ cache = ActiveTriples::Repositories.repositories[repository]
133
+ graph = RDF::Graph.new(:data => cache, :context => LinkedVocabs.vocabularies[name.to_sym][:source])
134
+ graph.load(LinkedVocabs.vocabularies[name.to_sym][:source])
135
+ graph
136
+ end
137
+
138
+ ##
139
+ # Implement QuestioningAuthority API
140
+ class QaRDF
141
+ attr_accessor :response, :raw_response
142
+
143
+ def initialize(parent=nil)
144
+ @parent = parent
145
+ end
146
+
147
+ ##
148
+ # Not a very smart sparql search. It's mostly intended to be
149
+ # overridden in subclasses, but it could also stand to be a bit
150
+ # better as a baseline RDF vocab search.
151
+ def search(q, sub_authority=nil)
152
+ @sparql = SPARQL::Client.new(ActiveTriples::Repositories.repositories[@parent.repository])
153
+ self.response = sparql_starts_search(q)
154
+ return response unless response.empty?
155
+ self.response = sparql_contains_search(q)
156
+ end
157
+
158
+ def results
159
+ response
160
+ end
161
+
162
+ def get_full_record(id, sub_authority)
163
+ end
164
+
165
+ private
166
+
167
+ def sparql_starts_search(q)
168
+ query = @sparql.query("SELECT DISTINCT ?s ?p ?o WHERE { ?s ?p ?o. FILTER(strstarts(lcase(?o), '#{q.downcase}'))}")
169
+ solutions_from_sparql_query(query)
170
+ end
171
+
172
+ def sparql_contains_search(q)
173
+ query = @sparql.query("SELECT DISTINCT ?s ?p ?o WHERE { ?s ?p ?o. FILTER(contains(lcase(?o), '#{q.downcase}'))}")
174
+ solutions_from_sparql_query(query)
175
+ end
176
+
177
+ def solutions_from_sparql_query(query)
178
+ # @TODO: labels should be taken from ActiveTriples::Resource.
179
+ # However, the default labels there are hidden behind a private method.
180
+ labels = [RDF::SKOS.prefLabel,
181
+ RDF::DC.title,
182
+ RDF::RDFS.label]
183
+ labels << @parent.rdf_label unless @parent.rdf_label.nil?
184
+
185
+ solutions = query.map { |solution| solution if @parent.uses_vocab_prefix? solution[:s] }.compact
186
+ label_solutions = solutions.map { |solution| build_hit(solution) if labels.include? solution[:p] }.compact
187
+ return label_solutions.uniq unless label_solutions.empty?
188
+ solutions.map { |solution| build_hit(solution) }.compact.uniq
189
+ end
190
+
191
+ def build_hit(solution)
192
+ { :id => solution[:s].to_s, :label => solution[:o].to_s }
193
+ end
194
+ end
195
+ end
196
+
197
+ class ControlledVocabularyError < StandardError; end
198
+ end
199
+ end