Mercurial > repos > guerler > springsuite
comparison planemo/lib/python3.7/site-packages/rdflib/extras/describer.py @ 1:56ad4e20f292 draft
"planemo upload commit 6eee67778febed82ddd413c3ca40b3183a3898f1"
author | guerler |
---|---|
date | Fri, 31 Jul 2020 00:32:28 -0400 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
0:d30785e31577 | 1:56ad4e20f292 |
---|---|
1 #!/usr/bin/env python | |
2 # -*- coding: utf-8 -*- | |
3 | |
4 from rdflib import py3compat | |
5 __doc__ = py3compat.format_doctest_out(""" | |
6 A Describer is a stateful utility for creating RDF statements in a | |
7 semi-declarative manner. It has methods for creating literal values, rel and | |
8 rev resource relations (somewhat resembling RDFa). | |
9 | |
10 The `rel` and ``rev`` methods return a context manager which sets the current | |
11 about to the referenced resource for the context scope (for use with the | |
12 ``with`` statement). | |
13 | |
14 Full example in the ``to_rdf`` method below:: | |
15 | |
16 >>> import datetime | |
17 >>> from rdflib.graph import Graph | |
18 >>> from rdflib.namespace import Namespace, RDFS, FOAF | |
19 >>> | |
20 >>> ORG_URI = "http://example.org/" | |
21 >>> | |
22 >>> CV = Namespace("http://purl.org/captsolo/resume-rdf/0.2/cv#") | |
23 >>> | |
24 >>> class Person(object): | |
25 ... def __init__(self): | |
26 ... self.first_name = %(u)s"Some" | |
27 ... self.last_name = %(u)s"Body" | |
28 ... self.username = "some1" | |
29 ... self.presentation = %(u)s"Just a Python & RDF hacker." | |
30 ... self.image = "/images/persons/" + self.username + ".jpg" | |
31 ... self.site = "http://example.net/" | |
32 ... self.start_date = datetime.date(2009, 9, 4) | |
33 ... def get_full_name(self): | |
34 ... return %(u)s" ".join([self.first_name, self.last_name]) | |
35 ... def get_absolute_url(self): | |
36 ... return "/persons/" + self.username | |
37 ... def get_thumbnail_url(self): | |
38 ... return self.image.replace('.jpg', '-thumb.jpg') | |
39 ... | |
40 ... def to_rdf(self): | |
41 ... graph = Graph() | |
42 ... graph.bind('foaf', FOAF) | |
43 ... graph.bind('cv', CV) | |
44 ... lang = 'en' | |
45 ... d = Describer(graph, base=ORG_URI) | |
46 ... d.about(self.get_absolute_url()+'#person') | |
47 ... d.rdftype(FOAF.Person) | |
48 ... d.value(FOAF.name, self.get_full_name()) | |
49 ... d.value(FOAF.firstName, self.first_name) | |
50 ... d.value(FOAF.surname, self.last_name) | |
51 ... d.rel(FOAF.homepage, self.site) | |
52 ... d.value(RDFS.comment, self.presentation, lang=lang) | |
53 ... with d.rel(FOAF.depiction, self.image): | |
54 ... d.rdftype(FOAF.Image) | |
55 ... d.rel(FOAF.thumbnail, self.get_thumbnail_url()) | |
56 ... with d.rev(CV.aboutPerson): | |
57 ... d.rdftype(CV.CV) | |
58 ... with d.rel(CV.hasWorkHistory): | |
59 ... d.value(CV.startDate, self.start_date) | |
60 ... d.rel(CV.employedIn, ORG_URI+"#company") | |
61 ... return graph | |
62 ... | |
63 >>> person_graph = Person().to_rdf() | |
64 >>> expected = Graph().parse(data='''<?xml version="1.0" encoding="utf-8"?> | |
65 ... <rdf:RDF | |
66 ... xmlns:foaf="http://xmlns.com/foaf/0.1/" | |
67 ... xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" | |
68 ... xmlns:cv="http://purl.org/captsolo/resume-rdf/0.2/cv#" | |
69 ... xmlns:rdfs="http://www.w3.org/2000/01/rdf-schema#"> | |
70 ... <foaf:Person rdf:about="http://example.org/persons/some1#person"> | |
71 ... <foaf:name>Some Body</foaf:name> | |
72 ... <foaf:firstName>Some</foaf:firstName> | |
73 ... <foaf:surname>Body</foaf:surname> | |
74 ... <foaf:depiction> | |
75 ... <foaf:Image | |
76 ... rdf:about= | |
77 ... "http://example.org/images/persons/some1.jpg"> | |
78 ... <foaf:thumbnail | |
79 ... rdf:resource= | |
80 ... "http://example.org/images/persons/some1-thumb.jpg"/> | |
81 ... </foaf:Image> | |
82 ... </foaf:depiction> | |
83 ... <rdfs:comment xml:lang="en"> | |
84 ... Just a Python & RDF hacker. | |
85 ... </rdfs:comment> | |
86 ... <foaf:homepage rdf:resource="http://example.net/"/> | |
87 ... </foaf:Person> | |
88 ... <cv:CV> | |
89 ... <cv:aboutPerson | |
90 ... rdf:resource="http://example.org/persons/some1#person"> | |
91 ... </cv:aboutPerson> | |
92 ... <cv:hasWorkHistory> | |
93 ... <rdf:Description> | |
94 ... <cv:startDate | |
95 ... rdf:datatype="http://www.w3.org/2001/XMLSchema#date" | |
96 ... >2009-09-04</cv:startDate> | |
97 ... <cv:employedIn rdf:resource="http://example.org/#company"/> | |
98 ... </rdf:Description> | |
99 ... </cv:hasWorkHistory> | |
100 ... </cv:CV> | |
101 ... </rdf:RDF> | |
102 ... ''') | |
103 >>> | |
104 >>> from rdflib.compare import isomorphic | |
105 >>> isomorphic(person_graph, expected) #doctest: +SKIP | |
106 True | |
107 """) | |
108 | |
109 from contextlib import contextmanager | |
110 from rdflib.graph import Graph | |
111 from rdflib.namespace import RDF | |
112 from rdflib.term import BNode | |
113 from rdflib.term import Identifier | |
114 from rdflib.term import Literal | |
115 from rdflib.term import URIRef | |
116 from rdflib.py3compat import format_doctest_out | |
117 | |
118 | |
119 class Describer(object): | |
120 | |
121 def __init__(self, graph=None, about=None, base=None): | |
122 if graph is None: | |
123 graph = Graph() | |
124 self.graph = graph | |
125 self.base = base | |
126 self._subjects = [] | |
127 self.about(about or None) | |
128 | |
129 @format_doctest_out | |
130 def about(self, subject, **kws): | |
131 """ | |
132 Sets the current subject. Will convert the given object into an | |
133 ``URIRef`` if it's not an ``Identifier``. | |
134 | |
135 Usage:: | |
136 | |
137 >>> d = Describer() | |
138 >>> d._current() #doctest: +ELLIPSIS | |
139 rdflib.term.BNode(...) | |
140 >>> d.about("http://example.org/") | |
141 >>> d._current() | |
142 rdflib.term.URIRef(%(u)s'http://example.org/') | |
143 | |
144 """ | |
145 kws.setdefault('base', self.base) | |
146 subject = cast_identifier(subject, **kws) | |
147 if self._subjects: | |
148 self._subjects[-1] = subject | |
149 else: | |
150 self._subjects.append(subject) | |
151 | |
152 @format_doctest_out | |
153 def value(self, p, v, **kws): | |
154 """ | |
155 Set a literal value for the given property. Will cast the value to an | |
156 ``Literal`` if a plain literal is given. | |
157 | |
158 Usage:: | |
159 | |
160 >>> from rdflib import URIRef | |
161 >>> from rdflib.namespace import RDF, RDFS | |
162 >>> d = Describer(about="http://example.org/") | |
163 >>> d.value(RDFS.label, "Example") | |
164 >>> d.graph.value(URIRef('http://example.org/'), RDFS.label) | |
165 rdflib.term.Literal(%(u)s'Example') | |
166 | |
167 """ | |
168 v = cast_value(v, **kws) | |
169 self.graph.add((self._current(), p, v)) | |
170 | |
171 @format_doctest_out | |
172 def rel(self, p, o=None, **kws): | |
173 """Set an object for the given property. Will convert the given object | |
174 into an ``URIRef`` if it's not an ``Identifier``. If none is given, a | |
175 new ``BNode`` is used. | |
176 | |
177 Returns a context manager for use in a ``with`` block, within which the | |
178 given object is used as current subject. | |
179 | |
180 Usage:: | |
181 | |
182 >>> from rdflib import URIRef | |
183 >>> from rdflib.namespace import RDF, RDFS | |
184 >>> d = Describer(about="/", base="http://example.org/") | |
185 >>> _ctxt = d.rel(RDFS.seeAlso, "/about") | |
186 >>> d.graph.value(URIRef('http://example.org/'), RDFS.seeAlso) | |
187 rdflib.term.URIRef(%(u)s'http://example.org/about') | |
188 | |
189 >>> with d.rel(RDFS.seeAlso, "/more"): | |
190 ... d.value(RDFS.label, "More") | |
191 >>> (URIRef('http://example.org/'), RDFS.seeAlso, | |
192 ... URIRef('http://example.org/more')) in d.graph | |
193 True | |
194 >>> d.graph.value(URIRef('http://example.org/more'), RDFS.label) | |
195 rdflib.term.Literal(%(u)s'More') | |
196 | |
197 """ | |
198 | |
199 kws.setdefault('base', self.base) | |
200 p = cast_identifier(p) | |
201 o = cast_identifier(o, **kws) | |
202 self.graph.add((self._current(), p, o)) | |
203 return self._subject_stack(o) | |
204 | |
205 @format_doctest_out | |
206 def rev(self, p, s=None, **kws): | |
207 """ | |
208 Same as ``rel``, but uses current subject as *object* of the relation. | |
209 The given resource is still used as subject in the returned context | |
210 manager. | |
211 | |
212 Usage:: | |
213 | |
214 >>> from rdflib import URIRef | |
215 >>> from rdflib.namespace import RDF, RDFS | |
216 >>> d = Describer(about="http://example.org/") | |
217 >>> with d.rev(RDFS.seeAlso, "http://example.net/"): | |
218 ... d.value(RDFS.label, "Net") | |
219 >>> (URIRef('http://example.net/'), RDFS.seeAlso, | |
220 ... URIRef('http://example.org/')) in d.graph | |
221 True | |
222 >>> d.graph.value(URIRef('http://example.net/'), RDFS.label) | |
223 rdflib.term.Literal(%(u)s'Net') | |
224 | |
225 """ | |
226 kws.setdefault('base', self.base) | |
227 p = cast_identifier(p) | |
228 s = cast_identifier(s, **kws) | |
229 self.graph.add((s, p, self._current())) | |
230 return self._subject_stack(s) | |
231 | |
232 def rdftype(self, t): | |
233 """ | |
234 Shorthand for setting rdf:type of the current subject. | |
235 | |
236 Usage:: | |
237 | |
238 >>> from rdflib import URIRef | |
239 >>> from rdflib.namespace import RDF, RDFS | |
240 >>> d = Describer(about="http://example.org/") | |
241 >>> d.rdftype(RDFS.Resource) | |
242 >>> (URIRef('http://example.org/'), | |
243 ... RDF.type, RDFS.Resource) in d.graph | |
244 True | |
245 | |
246 """ | |
247 self.graph.add((self._current(), RDF.type, t)) | |
248 | |
249 def _current(self): | |
250 return self._subjects[-1] | |
251 | |
252 @contextmanager | |
253 def _subject_stack(self, subject): | |
254 self._subjects.append(subject) | |
255 yield None | |
256 self._subjects.pop() | |
257 | |
258 | |
259 def cast_value(v, **kws): | |
260 if not isinstance(v, Literal): | |
261 v = Literal(v, **kws) | |
262 return v | |
263 | |
264 | |
265 def cast_identifier(ref, **kws): | |
266 ref = ref or BNode() | |
267 if not isinstance(ref, Identifier): | |
268 ref = URIRef(ref, **kws) | |
269 return ref |