Trying to play nice with SHACL

36 views
Skip to first unread message

Steve Ray

unread,
Nov 24, 2020, 12:40:00 PM11/24/20
to TopBraid Suite Users
I'm trying to get comfortable with using SHACL code and wean myself from embedded SPARQL. However, I'm having some trouble mapping certain common SPARQL patterns into the SHACL counterpart. Specifically, here's an example. I'm trying to find all the values of any property that is a subPropertyOf a parent property (c223:hasProperty).
The SPARQL should (I think) be something like this:

c223:PropertiesShape
  rdf:type sh:PropertyShape ;
  sh:path c223:hasProperty ;
  sh:name "PropertiesShape" ;
  sh:values [
      sh:sparql [
          sh:prefixes <http://www.w3.org/2000/01/rdf-schema> ;
          sh:select """SELECT DISTINCT ?prop
                             WHERE {
                             ?property rdfs:subPropertyOf* c223:hasProperty .
                             $this ?property ?prop .
}""" ;
        ] ;
    ] ;
.

I know the following is wrong, but not sure what sh: calls I should be using:

c223:PropertiesShape
  rdf:type sh:PropertyShape ;
  sh:path c223:hasProperty ;
  sh:name "PropertiesShape" ;
  sh:values [
      sh:distinct [
          sh:nodes [
              sh:path (
                  [
                    sh:zeroOrMorePath rdfs:subPropertyOf ;
                  ]
                  c223:hasProperty
                ) ;
            ] ;
        ] ;
    ] ;
.

This gives me a "malformed SHACL expression", so there are definitely problems.

My real question is, how does one duplicate this pattern in SHACL?:

WHERE {
                             ?property rdfs:subPropertyOf* <parentProperty> .
                             $this ?property ?result .
}




Steve


Irene Polikoff

unread,
Nov 24, 2020, 3:51:33 PM11/24/20
to topbrai...@googlegroups.com
There is no path in your example that could get from ?this to ?prop. A path specifies predicates. You do not have predicates to specify - you are finding out your predicates in the first WHERE statement. This is why you need to have 2 statements in the WHERE clause and can’t boil them down to a path expression in a single statement.

--
You received this message because you are subscribed to the Google Groups "TopBraid Suite Users" group.
To unsubscribe from this group and stop receiving emails from it, send an email to topbraid-user...@googlegroups.com.
To view this discussion on the web visit https://groups.google.com/d/msgid/topbraid-users/CAGUep87nmSVo5eziHPh6ExbPD1%3DD%3DWrfTNOwNtK0PBww21%3DpgA%40mail.gmail.com.

Steve Ray

unread,
Nov 24, 2020, 5:48:24 PM11/24/20
to TopBraid Suite Users
I agree with you. Is there no way to have "variable" predicates in native SHACL? Or is my only option here to use the embedded SPARQL?

Steve




Holger Knublauch

unread,
Nov 24, 2020, 5:57:52 PM11/24/20
to topbrai...@googlegroups.com


On 11/25/2020 8:48 AM, Steve Ray wrote:
I agree with you. Is there no way to have "variable" predicates in native SHACL? Or is my only option here to use the embedded SPARQL?

Exactly. SHACL node expressions do not have a notion of variables. It is intentionally limited to be easier, but then doesn't offer all features that SPARQL does. There is nothing wrong with using SPARQL node expressions as a fallback.

Holger


Steve Ray

unread,
Nov 24, 2020, 9:18:21 PM11/24/20
to TopBraid Suite Users
Sorry to keep using up bandwidth here. I have pored over all the documentation I can find on the proper use of sh:values with SPARQL, and clearly I'm not getting something. Here is my trivial toy example that is supposed to just infer a hard-coded value of "Hello" for a property. Can you see what I'm doing wrong?
TBC keeps saying: Property ssh:Property_1: values=(SHACL node expression of unknown type: java.lang.IllegalArgumentException: Malformed SHACL node expression)

 I'm also attaching the full file in case you want to just load it.

ssh:Class_1
a owl:Class ;
a sh:NodeShape ;
rdfs:subClassOf owl:Thing ;
sh:property [
sh:path ssh:Property_1 ;
sh:values [
sh:sparql [
sh:select """SELECT ?result
WHERE {
BIND (\"Hello\" AS ?result) .
}""" ;
] ;
] ;
] ;
.
ssh:Instance_1
a ssh:Class_1 ;
.
ssh:Property_1
a rdf:Property ;
.



Steve




sparqlshape.shapes.ttl

Holger Knublauch

unread,
Nov 24, 2020, 9:25:53 PM11/24/20
to topbrai...@googlegroups.com


On 11/25/2020 12:18 PM, Steve Ray wrote:
Sorry to keep using up bandwidth here.

No problem. Admittedly, the official spec at https://w3c.github.io/shacl/shacl-af/#select is not exactly full of examples. A failure of the editor :)


I have pored over all the documentation I can find on the proper use of sh:values with SPARQL, and clearly I'm not getting something. Here is my trivial toy example that is supposed to just infer a hard-coded value of "Hello" for a property. Can you see what I'm doing wrong?
TBC keeps saying: Property ssh:Property_1: values=(SHACL node expression of unknown type: java.lang.IllegalArgumentException: Malformed SHACL node expression)

 I'm also attaching the full file in case you want to just load it.

Just drop the sh:sparql node and instead do something like

sh:values [
    sh:select ...
    sh:prefixes ...
]

HTH
Holger


Steve Ray

unread,
Nov 24, 2020, 9:32:45 PM11/24/20
to TopBraid Suite Users
Wow, fast!
I was just trying exactly that. I also dropped the prefixes since I wasn't using any. So now I have:

ssh:Class_1

  rdf:type owl:Class ;

  rdf:type sh:NodeShape ;

  rdfs:subClassOf owl:Thing ;

  sh:property [

      sh:path ssh:Property_1 ;

      sh:values [

          sh:select """SELECT ?result

                           WHERE {

                           BIND (\"Hello\" AS ?result) .

                           }""" ;

        ] ;

    ] ;

.


But executing this query:


SELECT *

WHERE {

?subject a ssh:Class_1 .

    ?subject ssh:Property_1 ?object .

}


...is coming up empty.


(Once I get one example working, I'm off to the races!)


Steve




Holger Knublauch

unread,
Nov 24, 2020, 9:43:48 PM11/24/20
to topbrai...@googlegroups.com


On 11/25/2020 12:32 PM, Steve Ray wrote:
Wow, fast!
I was just trying exactly that. I also dropped the prefixes since I wasn't using any. So now I have:

ssh:Class_1

  rdf:type owl:Class ;

  rdf:type sh:NodeShape ;

  rdfs:subClassOf owl:Thing ;

  sh:property [

      sh:path ssh:Property_1 ;

      sh:values [

          sh:select """SELECT ?result

                           WHERE {

                           BIND (\"Hello\" AS ?result) .

                           }""" ;

        ] ;

    ] ;

.

So far so good.


But executing this query:


SELECT *

WHERE {

?subject a ssh:Class_1 .

    ?subject ssh:Property_1 ?object .

}


...is coming up empty.

This is because SPARQL only queries the asserted graph. To query the inferred values, either use GraphQL, Active Data Shapes or use this magic property:

PREFIX teamwork: <http://topbraid.org/teamwork#>
SELECT *
WHERE {
  ...
  (?subject ssh:Property_1 <urn:x-evn-master:geo>) teamwork:values ?object .
}

This takes the master graph that is holding the data in EDG (here: the Geography ontology). This is the fastest variation for performance, but this one here is more general:

  (?subject ssh:Property_1) tosh:values ?object .

This design has some implications, e.g. you normally cannot query inferred values in the inverse direction, e.g. if ?object is given. They are just computed on the fly, for the duration of the query. This means that the system doesn't need to keep track of inferences or invalidate them if the data changes - it's always up to date.

Holger


Steve Ray

unread,
Nov 24, 2020, 9:56:28 PM11/24/20
to TopBraid Suite Users
Oh dear. I'm doing this work in support of a standards committee where I'm the only one using TBC. Others are using either Jena, or raw text editors and home-rolled reasoners. (!) (I may be persuading one member to get TBC though).

Of the solutions you are suggesting, which would you recommend that is the most universal? Does Jena support SPIN (i.e. magic properties?). I'm assuming tosh isn't available outside TBC/EDG?

Steve




Holger Knublauch

unread,
Nov 24, 2020, 10:51:18 PM11/24/20
to topbrai...@googlegroups.com

I cannot think of anything in this inferencing area (with SPARQL-like expressiveness) that is fully standardized or widely available.

Holger

Reply all
Reply to author
Forward
0 new messages