Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SHACL processor #236

Merged
merged 69 commits into from
Jun 30, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
69 commits
Select commit Hold shift + click to select a range
e62bfdd
Adds SHACL functionality and test suite.
langsamu Mar 11, 2019
f3052a4
Adds datatype constraint.
langsamu Mar 11, 2019
ea058b4
Adds and constraint.
langsamu Mar 12, 2019
623ad71
Adds or constraint.
langsamu Mar 12, 2019
792e112
Adds not constraint.
langsamu Mar 12, 2019
c6095ec
Adds xone constraint.
langsamu Mar 12, 2019
06cf091
Adds node kind constraint.
langsamu Mar 12, 2019
95a388c
Adds min length constraint.
langsamu Mar 12, 2019
2a34dfb
Adds max length constraint.
langsamu Mar 12, 2019
7106de0
Adds langguage match constraint.
langsamu Mar 12, 2019
37cdafb
Adds in constraint.
langsamu Mar 12, 2019
695da4e
Gives constraints access to all focus nodes.
langsamu Mar 13, 2019
a4a25e6
Adds min count constraint.
langsamu Mar 13, 2019
0f5d949
Adds max count constraint.
langsamu Mar 13, 2019
a2a5506
Fixes node target.
langsamu Mar 13, 2019
ef4b7e8
Adds unique language constraint.
langsamu Mar 13, 2019
b1c6e6f
Adds has value constraint.
langsamu Mar 13, 2019
abcc31a
Fixes class target.
langsamu Mar 13, 2019
c3f34ca
Gives constraints access to their triple, not just their object.
langsamu Mar 13, 2019
bba4de0
Adds pattern constraint.
langsamu Mar 13, 2019
4e601df
Gives constraints access to focus node in addition to value nodes.
langsamu Mar 14, 2019
4cf9fcf
Adds equals constraint.
langsamu Mar 14, 2019
c2e3ea6
Adds disjoint constraint.
langsamu Mar 14, 2019
6beb68a
Adds less than constraint.
langsamu Mar 15, 2019
2d358ec
Adds less than or equals constraint.
langsamu Mar 15, 2019
1bda04c
Adds min exclusive constraint.
langsamu Mar 15, 2019
ee1d0b5
Adds min inclusive constraint.
langsamu Mar 15, 2019
a81ee51
Adds max exclusive constraint.
langsamu Mar 15, 2019
4710bf8
Adds max inclusive constraint.
langsamu Mar 15, 2019
86f9a3a
Fixes min count constraint to handle 0.
langsamu Mar 15, 2019
38f2364
Adds qualified count constraints.
langsamu Mar 16, 2019
4f2692f
Adds closed constraint.
langsamu Mar 16, 2019
da97ba3
Fixes class constraint so it doesn't depend on value nodes.
langsamu Mar 16, 2019
585864e
Adds validation report plumbing.
langsamu Mar 17, 2019
34cd91f
Implements validation report for 8 constarints.
langsamu Mar 19, 2019
18be2a8
Implements validation report for 9 constarints.
langsamu Mar 20, 2019
6cf105a
Compare constraint hierarchy.
langsamu Mar 20, 2019
3e6bbd7
Numeric constraint hierarcy.
langsamu Mar 20, 2019
4aee030
Boolean constraint hierarchy.
langsamu Mar 20, 2019
3db0369
Adds result path internals to validation report.
langsamu Mar 21, 2019
b0b4a9f
Implements literal wellformdness checking for datatype constraint.
langsamu Mar 25, 2019
c1ed185
Fixes value range constraints.
langsamu Mar 25, 2019
15fe553
Fixes uniqu language constraint.
langsamu Mar 25, 2019
d2c8d13
Implements report comparison in test suite.
langsamu Mar 25, 2019
7aa5aca
Almost full compliance with SHACL Core .
langsamu Mar 27, 2019
562ee79
Enable node has value test (see w3c/data-shapes#111).
langsamu Mar 31, 2019
1fa66d0
Adds bnode rewriting to report describe algorithm.
langsamu Mar 31, 2019
fc148f6
Adds SPARQL constraint.
langsamu Apr 2, 2019
066781c
Adds optional pre-bound variables to sparql constraint.
langsamu Apr 2, 2019
90de22d
Fails invalid queries in sparql constraint.
langsamu Apr 2, 2019
968c2b7
Implements SPARQL-based constraint components.
langsamu Apr 5, 2019
c2a348a
Adds value node binding to ask validator.
langsamu Apr 5, 2019
98f45f9
Fixes constraint delegation for components.
langsamu Apr 6, 2019
966286a
Tidies up SHACL test suite.
langsamu Apr 7, 2019
45153ae
Sparql constraint hierarchy.
langsamu Apr 9, 2019
9edf960
Adds SHACL examle test.
langsamu Apr 10, 2019
579fc39
Reduces SHACL API surface.
langsamu Apr 11, 2019
a31312e
Removes SHACL prefixes and suffixes.
langsamu Apr 11, 2019
e241c4b
Moves all nodes known to SHACL into vocabulary class.
langsamu Apr 11, 2019
3fd3684
Merges master (mostly Dynamic API #228) into Shacl processor.
langsamu Apr 12, 2019
5de6144
Opens SHACL test suite data so it can be used for report generation.
langsamu Apr 21, 2019
9f7a6d6
Merges master, removes woraround for #237 fixed by #245.
langsamu Apr 22, 2019
82d5eea
Improves code style and some naming.
langsamu May 9, 2019
0e96215
Merge branch 'master' of https://github.com/dotnetrdf/dotnetrdf into …
langsamu May 9, 2019
4330e0a
Excludes tests with datatype contraints from NETCORE target.
langsamu May 9, 2019
f3538c0
Datatype constraint builds schema programmatically.
langsamu May 10, 2019
f873b18
Removes workaround for #233 fixed by #234.
langsamu May 10, 2019
db335d4
Adds documentation comments.
langsamu May 13, 2019
aca4668
Adds SHACL implementation report generator.
langsamu May 14, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
74 changes: 73 additions & 1 deletion Libraries/dotNetRDF/Core/WrapperNode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,19 +27,21 @@
namespace VDS.RDF
{
using System;
using System.Diagnostics;
using VDS.RDF.Writing;
using VDS.RDF.Writing.Formatting;

/// <summary>
/// Abstract decorator for Nodes to make it easier to layer functionality on top of existing implementations.
/// </summary>
public abstract partial class WrapperNode : INode
public abstract partial class WrapperNode : INode, IBlankNode, IUriNode, ILiteralNode
{
/// <summary>
/// Initializes a new instance of the <see cref="WrapperNode"/> class.
/// </summary>
/// <param name="node">The node this is a wrapper around.</param>
/// <exception cref="ArgumentNullException">When <paramref name="node"/> is null.</exception>
[DebuggerStepThrough]
protected WrapperNode(INode node)
{
Node = node ?? throw new ArgumentNullException(nameof(node));
Expand Down Expand Up @@ -77,6 +79,76 @@ public Uri GraphUri
}
}

/// <inheritdoc/>
string IBlankNode.InternalID
{
get
{
if (Node.NodeType != NodeType.Blank)
{
throw new InvalidCastException();
}

return ((IBlankNode)Node).InternalID;
}
}

/// <inheritdoc/>
Uri IUriNode.Uri
{
get
{
if (Node.NodeType != NodeType.Uri)
{
throw new InvalidCastException();
}

return ((IUriNode)Node).Uri;
}
}

/// <inheritdoc/>
string ILiteralNode.Value
{
get
{
if (Node.NodeType != NodeType.Literal)
{
throw new InvalidCastException();
}

return ((ILiteralNode)Node).Value;
}
}

/// <inheritdoc/>
string ILiteralNode.Language
{
get
{
if (Node.NodeType != NodeType.Literal)
{
throw new InvalidCastException();
}

return ((ILiteralNode)Node).Language;
}
}

/// <inheritdoc/>
Uri ILiteralNode.DataType
{
get
{
if (Node.NodeType != NodeType.Literal)
{
throw new InvalidCastException();
}

return ((ILiteralNode)Node).DataType;
}
}

/// <summary>
/// Gets the underlying node this is a wrapper around.
/// </summary>
Expand Down
226 changes: 226 additions & 0 deletions Libraries/dotNetRDF/Shacl/Constraint.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
/*
// <copyright>
// dotNetRDF is free and open source software licensed under the MIT License
// -------------------------------------------------------------------------
//
// Copyright (c) 2009-2017 dotNetRDF Project (http://dotnetrdf.org/)
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is furnished
// to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
// </copyright>
*/

namespace VDS.RDF.Shacl
{
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using VDS.RDF.Shacl.Constraints;
using VDS.RDF.Shacl.Validation;

internal abstract class Constraint : WrapperNode
{
[DebuggerStepThrough]
protected Constraint(Shape shape, INode value)
: base(value)
{
this.Shape = shape;
}

internal abstract INode ConstraintComponent { get; }

protected Shape Shape { get; private set; }

internal static Constraint Parse(Shape shape, INode type, INode value)
{
switch (type)
{
case INode t when t.Equals(Vocabulary.Property): return new Property(shape, value);
case INode t when t.Equals(Vocabulary.MaxCount): return new MaxCount(shape, value);
case INode t when t.Equals(Vocabulary.NodeKind): return new NodeKind(shape, value);
case INode t when t.Equals(Vocabulary.MinCount): return new MinCount(shape, value);
case INode t when t.Equals(Vocabulary.Node): return new Node(shape, value);
case INode t when t.Equals(Vocabulary.Datatype): return new Datatype(shape, value);
case INode t when t.Equals(Vocabulary.Closed): return new Closed(shape, value);
case INode t when t.Equals(Vocabulary.HasValue): return new HasValue(shape, value);
case INode t when t.Equals(Vocabulary.Or): return new Or(shape, value);
case INode t when t.Equals(Vocabulary.Class): return new Class(shape, value);
case INode t when t.Equals(Vocabulary.Not): return new Not(shape, value);
case INode t when t.Equals(Vocabulary.Xone): return new Xone(shape, value);
case INode t when t.Equals(Vocabulary.In): return new In(shape, value);
case INode t when t.Equals(Vocabulary.Sparql): return new Select(shape, value);
case INode t when t.Equals(Vocabulary.Pattern): return new Pattern(shape, value);
case INode t when t.Equals(Vocabulary.MinInclusive): return new MinInclusive(shape, value);
case INode t when t.Equals(Vocabulary.MinExclusive): return new MinExclusive(shape, value);
case INode t when t.Equals(Vocabulary.MaxExclusive): return new MaxExclusive(shape, value);
case INode t when t.Equals(Vocabulary.MinLength): return new MinLength(shape, value);
case INode t when t.Equals(Vocabulary.MaxInclusive): return new MaxInclusive(shape, value);
case INode t when t.Equals(Vocabulary.And): return new And(shape, value);
case INode t when t.Equals(Vocabulary.QualifiedMinCount): return new QualifiedMinCount(shape, value);
case INode t when t.Equals(Vocabulary.QualifiedMaxCount): return new QualifiedMaxCount(shape, value);
case INode t when t.Equals(Vocabulary.EqualsNode): return new Equals(shape, value);
case INode t when t.Equals(Vocabulary.LanguageIn): return new LanguageIn(shape, value);
case INode t when t.Equals(Vocabulary.LessThan): return new LessThan(shape, value);
case INode t when t.Equals(Vocabulary.Disjoint): return new Disjoint(shape, value);
case INode t when t.Equals(Vocabulary.LessThanOrEquals): return new LessThanOrEquals(shape, value);
case INode t when t.Equals(Vocabulary.UniqueLang): return new UniqueLang(shape, value);
case INode t when t.Equals(Vocabulary.MaxLength): return new MaxLength(shape, value);

default: throw new Exception();
}
}

internal abstract bool Validate(INode focusNode, IEnumerable<INode> valueNodes, Report report);

protected bool ReportValueNodes(INode focusNode, IEnumerable<INode> invalidValues, Report report)
{
var allValid = !invalidValues.Any();

if (report is null)
{
return allValid;
}

if (allValid)
{
return true;
}

foreach (var invalidValue in invalidValues)
{
var result = Result.Create(report.Graph);
result.SourceConstraintComponent = ConstraintComponent;
result.Severity = Shape.Severity;
result.Message = Shape.Message;
result.SourceShape = Shape;
result.FocusNode = focusNode;

if (Shape is Shapes.Property propertyShape)
{
result.ResultPath = propertyShape.Path;
report.Graph.Assert(propertyShape.Path.AsTriples.Select(t => t.CopyTriple(report.Graph)));
}

result.ResultValue = invalidValue;

report.Results.Add(result);
}

return false;
}

protected bool ReportValueNodes(IEnumerable<Triple> invalidValues, Report report)
{
var allValid = !invalidValues.Any();

if (report is null)
{
return allValid;
}

if (allValid)
{
return true;
}

foreach (var invalidValue in invalidValues)
{
var result = Result.Create(report.Graph);
result.SourceConstraintComponent = ConstraintComponent;
result.Severity = Shape.Severity;
result.Message = Shape.Message;
result.SourceShape = Shape;
result.FocusNode = invalidValue.Subject;
result.ResultPath = Path.Parse(invalidValue.Predicate);
report.Graph.Assert(result.ResultPath.AsTriples.Select(t => t.CopyTriple(report.Graph)));
result.ResultValue = invalidValue.Object;

report.Results.Add(result);
}

return false;
}

protected bool ReportFocusNode(INode focusNode, IEnumerable<INode> invalidValues, Report report)
{
var allValid = !invalidValues.Any();

if (report is null)
{
return allValid;
}

if (allValid)
{
return true;
}

var result = Result.Create(report.Graph);
result.SourceConstraintComponent = ConstraintComponent;
result.Severity = Shape.Severity;
result.Message = Shape.Message;
result.SourceShape = Shape;
result.FocusNode = focusNode;

if (Shape is Shapes.Property propertyShape)
{
result.ResultPath = propertyShape.Path;
report.Graph.Assert(propertyShape.Path.AsTriples.Select(t => t.CopyTriple(report.Graph)));
}

report.Results.Add(result);

return false;
}

protected bool ReportFocusNodes(INode focusNode, IEnumerable<INode> invalidValues, Report report)
{
var allValid = !invalidValues.Any();

if (report is null)
{
return allValid;
}

if (allValid)
{
return true;
}

foreach (var invalidValue in invalidValues)
{
var result = Result.Create(report.Graph);
result.SourceConstraintComponent = ConstraintComponent;
result.Severity = Shape.Severity;
result.Message = Shape.Message;
result.SourceShape = Shape;
result.FocusNode = focusNode;

if (Shape is Shapes.Property propertyShape)
{
result.ResultPath = propertyShape.Path;
report.Graph.Assert(propertyShape.Path.AsTriples.Select(t => t.CopyTriple(report.Graph)));
}

report.Results.Add(result);
}

return false;
}
}
}
Loading